《UNIX网络编程 卷1》 笔记: readn和writen函数

字节流套接字上调用read或write读取或写入的字节数可能比请求的数量少,这是正常的现象。原因在于内核中用于套接字的缓冲区可能已达到了极限(例如套接字的接收缓冲区可能没有数据或发送缓冲区没有可用空间),此时所需的是调用者再次调用read或write函数,以读取或写入剩余的字节。

我们实现的从套接字上读取n字节数据的readn函数和写入n字节数据到套接字的writen函数代码如下:

 

#include "unp.h"

/*从套接字中读取n字节数据*/
ssize_t readn(int fd, void *vptr, size_t n)
{
	size_t nleft;
	ssize_t nread;
	char *ptr;

	ptr = vptr;
	nleft = n;
	while (nleft > 0) {
		if ((nread = read(fd, ptr, nleft)) < 0) {
			if (errno == EINTR) /*重试*/
				nread = 0;
			else  /*错误*/
				return -1;
		} else if (nread == 0) /*遇到EOF*/
			break;
		nleft -= nread;
		ptr += nread;
	}

	return n - nleft;
}

/*包裹函数*/
ssize_t Readn(int fd, void *vptr, size_t n)
{
	if (readn(fd, vptr, n) == -1)
		err_sys("readn error");
}

/*写入n字节数据到套接字*/
ssize_t writen(int fd, const void *vptr, size_t n)
{
    size_t nleft;
    ssize_t nwriten;
    const char *ptr;

    ptr = vptr;
    nleft = n;
    while (nleft > 0) {
        if ((nwriten = write(fd, ptr, nleft)) <= 0) {
            if (nwriten < 0 && errno == EINTR) /*重试*/
                nwriten = 0;
            else /*错误*/
                return -1;
        }
        nleft -= nwriten;
        ptr += nwriten;
    }

    return n;
}

/*包裹函数*/
ssize_t Writen(int fd, const void *vptr, size_t n)
{
    if (writen(fd, vptr, n) == -1)
        err_sys("writen error");
} 

 

注:read或write系统调用在执行过程中被信号打断时会产生EINTR错误,这时应该忽略这个错误并继续被中断的系统调用。

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值