读书随笔_2016.3.18

1.fprint函数

函数声明:int fprintf(FILE * stream, const char * format, ...);

函数说明:fprintf()会根据参数format 字符串来转换并格式化数据, 然后将结果输出到参数stream 指定的文件中, 直到出现字符串结束('\0')为止。

返回值:关于参数format 字符串的格式请参考printf(). 成功则返回实际输出的字符数, 失败则返回-1, 错误原因存于errno 中.

#include <stdio.h>

int main()
{
	int i=1;
	char *s="cyf";
	fprintf(stdout,"%s,%d",s,i);

	return 0;
}


2.vsprintf函数

函数声明:int vfprintf(FILE *stream, char *format, va_list param);

函数说明:发送格式化输出到一个流使用传递给它的参数列表。

返回值:如果成功,返回写入的字符的总数,否则则返回一个负数。

#include <stdio.h>
#include <stdarg.h>
#include <time.h>
#include <stdlib.h>
#include <string.h>
#define debug(...)\
    _debug(__FILE__, __LINE__, __VA_ARGS__)
void _debug(const char *file, int line, const char *fmt, ...)
{
    va_list argp;
    time_t timep;
    char *timestr;
    struct tm *local;

    time(&timep);
    local=localtime(&timep);
    timestr=asctime(local);

    fprintf(stderr,"[%.*s] %s:%d ",(int)(strlen(timestr)-1), timestr, file, line);

    va_start(argp,fmt);
    vfprintf(stderr, fmt, argp);
    fflush(stderr);
    va_end(argp);

}

int main()
{
    int i=1;

    if(i!=2)
    {
        debug("the number is not %d",i);
    }
    return 0;
}


自定义调试信息的输出

  调试信息的输出方法有很多种,  例如直接用printf,  或者出错时使用perror, fprintf等将信息直接打印到终端上, 在Qt上面一般使用qDebug,而守护进程则一般是使用syslog将调试信息输出到日志文件中等等...
  使用标准的方法打印调试信息有时候不是很方便,  例如Qt编程, 在调试已有的代码时, 我想在打印调试信息的地方, 把代码位置也打印出来以方便定位错误, 或者需要在调试信息前面加一个前辍, 好方便在调试信息太多的时候可以用grep过滤一下, 仅显示本模块的调试信息, 这时就需要一个一个地修改已有的qDebug, 使其成为以下形式:
  qDebug( "[模块名称] 调试信息  File:%s, Line:%d", __FILE__, __LINE__ );
  这样的修改比较烦人, 而且一不小心会遗漏某个没改的...
  为了能方便地管理调试信息的输出,一个比较简单的方法就是自已定义一个打印调试信息的宏, 然后替换原来的,废话就不多说了,直接给出一个现成的,下面是一个例子, 我用WiFi表示当前代码的模块名称,我要求在模块中的所有调试信息前面均带有[WiFi]前辍,这样我就能方便地只需使用命令行 | grep "\[WiFi\]"来过滤掉来自其它模块的调试信息了:
#define qWiFiDebug(format, ...) qDebug("[WiFi] "format" File:%s, Line:%d, Function:%s", ##__VA_ARGS__, __FILE__, __LINE__ , __FUNCTION__);
  上面的宏是使用qDebug输出调试信息,在非Qt的程序中也可以改为printf,守护进程则可以改为syslog等等...  其中,决窍其实就是这几个宏 ##__VA_ARGS__, __FILE__, __LINE__ 和__FUNCTION__,下面介绍一下这几个宏:
  1)  __VA_ARGS__ 是一个可变参数的宏,很少人知道这个宏,这个可变参数的宏是新的C99规范中新增的,目前似乎只有gcc支持(VC6.0的编译器不支持)。宏前面加上##的作用在于,当可变参数的个数为0时,这里的##起到把前面多余的","去掉的作用,否则会编译出错, 你可以试试。
  2) __FILE__ 宏在预编译时会替换成当前的源文件名
  3) __LINE__宏在预编译时会替换成当前的行号
  4) __FUNCTION__宏在预编译时会替换成当前的函数名称
  有了以上这几个宏,特别是有了__VA_ARGS__ ,调试信息的输出就变得灵活多了。

3.c语言%.*s是什么

*用来指定宽度,对应一个整数
.(点)与后面的数合起来 是指定必须输出这个宽度,如果所输出的字符串长度大于这个数,则按此宽度输出,如果小于,则输出实际长度

4.memset

函数声明:void *memset(void *s, int ch, size_t n);

用途:为一段内存的每一个字节都赋予ch所代表的值,该值采用ASCII编码。

所属函数库:<memory.h> 或者 <string.h>

#include <stdio.h>
#include <string.h>

int main()
{
	int a[10];

	memset(a,'\0',sizeof(a));

	int i;

	for(i=0;i<10;i++)
		printf("%d ",a[i]);
	
	return 0;
}

5.sprintf

函数声明:int sprintf(char *str, const char *format, ...)

用途:把格式化的数据写入到某个字符串

#include <stdio.h>

int main()
{
	char buff[10];
	char *s="this is not ok";
	sprintf(buff,"%s %d", s, 10);

	printf("%s",buff);

	return 0;
}
6.fgets

函数声明:char *fgets(char *s, int n, FILE *stream);

参数:

s:字符行指针,指向存储读入数据的缓冲去的地址

n:从流中读入n-1个字符

stream:指向读取的流


strchr

函数声明:char *strchr(char *str, char ch)

用途:找出在资付出str中第一次出现字符ch的位置,并返回该字符位置的指针,找不到就返回空指针

strstr

#include <stdio.h>
#include <string.h>
#define DEFAUT_FILE "/home/cyf/tsar/conf/tsar.conf"

int main()
{
	FILE *fp;
	char buff[1024];

	if(!(fp=fopen(DEFAUT_FILE,"r")))
	{
		fprintf(stderr,"%s","error in start");
	}
	while(fgets(buff,1024,fp))
	{
		char *temp;

		if(temp=strchr(buff,'\n'))
			*temp='\0';
		if(temp=strchr(buff,'\r'))
			*temp='\0';
		if(buff[0]=='#')
			memset(buff, '\0', 1024);
		printf("%s",buff);
	}
	if(fclose(fp)<0)
		fprintf(stderr,"%s","error");

	return 0;
}


函数原型:char *strstr( char *str, char * substr );
【参数说明】str为要检索的字符串,substr为要检索的子串。

【返回值】返回字符串str中第一次出现子串substr的地址;如果没有检索到子串,则返回NULL。



你可能感兴趣的:(读书随笔_2016.3.18)