APUE看了有一段时间了,最初开始看的时候都是用printf函数代替了err_xxx等错误处理函数,然后作者中的apue.h头文件我并没有用,而是用man查找函数所在的头文件,然后写到程序中,这样不用下载作者的源码,自己也可以实现书上的例子。如今有时间来看了书上附录B的出错处理函数的部分,发现也不是很难,反而又获得了一些新知识。其实我们可以把这些错误处理函数编译成动态链接库,这样以后就可以直接调用了。下面来讲一下出错处理函数中出现一些知识点。其实作者的出错处理函数就是一个可变参数的函数
例如有一个可变参数的函数声明如下:void func(char *fmt,…);//三个点表示可变参数
va_start(ap,fmt)表示得到可变参数中的第一个参数的地址即函数func中…所代表的第一个参数的地址,并把这个地址赋给ap,ap的类型为va_list
p=va_arg(ap,type)表示下一个参数的地址给ap,并返回当前参数的值给p,其中怕的类型就是va_arg(ap,type)传入的type.
va_end清空va_list变量
至于这些宏是如何实现的,我综合了网上的几篇文章,感觉写的还比较好。链接如下
http://blog.csdn.net/baidu_24256693/article/details/43677095
然后是我自己是现的类printf函数简单实现
程序是用vim写的,用tab敲的空格可能看起来有点长
#include<unistd.h>
#include<stdarg.h>
#include<stdio.h>
#include<string.h>
#include<malloc.h>
void myprintf(const char *fmt,...)
{
va_list ap;
char str[1024];
char *strp=str;
char s[10];
char *sp;
int len;
int i=0;
va_start(ap,fmt);
do
{
if(*fmt!='%')
{
*strp++=*fmt;
}
else
{
switch(*++fmt)
{
case 'd':
sp=s;
len=va_arg(ap,int);
while(len)
{
*sp++=len%10+'0';
len/=10;
}
len=strlen(s)-1;
for(i=0;i<len;i++) *strp++=*(--sp);
break;
case 'c': len=va_arg(ap,int);
*strp++=len;
break;
case 's':
sp=va_arg(ap,char *);
while(*sp!=0)
{
*strp++=*sp++;
}
break;
default :break;
}
}
++fmt;
}while(*fmt!=0);
*strp=0;
va_end(ap);
puts(str);
return ;
}
int main()
{
myprintf("i=%d,j=%d",100,200);
myprintf("string=%s,character=%c","str",'c');
return 0;
}