Buffer的实现
include
struct iovec {
ptr_t iov_base; // Starting address
size_t iov_len; //Length in bytes
};
struct iovec定义了一个向量元素。通常,这个结构用作一个多元素的数组。
对于每一个传输的元素,指针成员iov_base指向一个缓冲区,
这个缓冲区是存放的是readv所接收的数据或是writev将要发送的数据。
成员iov_len在各种情况下分别确定了接收的最大长度以及实际写入的长度。
*/
/*
为什么引出readv()和writev()
因为使用read()将数据读到不连续的内存、使用write()将不连续的内存发送出去,要经过多次的调用read、write
如果要从文件中读一片连续的数据至进程的不同区域,有两种方案:①使用read()一次将它们读至一个较大的缓冲区中,然后将它们分成若干部分复制到不同的区域; ②调用read()若干次分批将它们读至不同区域。
同样,如果想将程序中不同区域的数据块连续地写至文件,也必须进行类似的处理。
怎么解决多次系统调用+拷贝带来的开销呢?
UNIX提供了另外两个函数—readv()和writev(),它们只需一次系统调用就可以实现在文件和进程的多个缓冲区之间传送数据,免除了多次系统调用或复制数据的开销。
readv/writev
在一次函数调用中:
① writev以顺序iov[0]、iov[1]至iov[iovcnt-1]从各缓冲区中聚集输出数据到fd
② readv则将从fd读入的数据按同样的顺序散布到各缓冲区中,readv总是先填满一个缓冲区,然后再填下一个
include
ssize_t readv(int fd, const struct iovec iov, int iovcnt);
ssize_t writev(int fd, const struct iovec iov, int iovcnt);
————————————————
*/
ssize_t readv(int fd, const struct iovec iov, int iovcnt);
ssize_t writev(int fd, const struct iovec iov, int iovcnt);
————————————————
*/
日志的宏定义
/*
一、#用来把参数转换成字符串
例子一:
#define P(A) printf(“%s:%d\n”,#A,A);
int main(int argc, char **argv)
{
int a = 1, b = 2;
P(a);
P(b);
P(a+b);
system(“pause”);
}
输出为:
例子二:
define SQUARE(x) printf(“The square of “#x” is %d.\n”, ((x)*(x)));
SQUARE(8)
输出的是:The square of 8 is 64
二、##运算符可以用于宏函数的替换部分。这个运算符把两个语言符号组合成单个语言符号,
为宏扩展提供了一种连接实际变元的手段
例子一:
#define XNAME(n) x ## n
如果这样使用宏:XNAME(8)
则会被展开成这样:x8
就是个粘合剂,将前后两部分粘合起来,也就是有“字符化”的意思。但是“##”不能随意粘合任意字符,必须是合法的C语言标示符。在单一的宏定义中,最多可以出现一次“#”或“##”预处理操作符。如果没有指定与“#”或“##”预处理操作符相关的计算次序,则会产生问题。为避免该问题,在单一的宏定义中只能使用其中一种操作符(即,一份“#”或一个“##”,或都不用)。除非非常有必要,否则尽量不要使用“#”和“##”。
三、VA_ARGS 是一个可变参数的宏,很少人知道这个宏,这个可变参数的宏是新的C99规范中新增的,目前似乎只有gcc支持(VC6.0的编译器不支持)。
实现思想就是宏定义中参数列表的最后一个参数为省略号(也就是三个点)。
四、##VA_ARGS 宏前面加上##的作用在于,当可变参数的个数为0时,这里的##起到把前面多余的”,”去掉的作用,否则会编译出错
一般这个用在调试信息上多一点
例如:
#define my_print1(…) printf(VA_ARGS) my_print1(“i=%d,j=%d\n”,i,j) 正确打印
#define my_print2(fmt,…) printf(fmt,VA_ARGS)
my_print1(“i=%d,j=%d\n”,i,j) 正确打印
my_print2(“iiiiiii\n”) 编译失败打印,因为扩展出来只有一个参数,至少要两个及以上参数
如果是#define my_print2(fmt,…) printf(fmt,##VA_ARGS)
那么
my_print1里面不管是几个参数都能正确打印
*/