socket编程中recv()和read()的使用与区别

recv和read相似,都可用来接收sockfd发送的数据,但recv比read多了一个参数,也就是第四个参数,它可以指定标志来控制如何接收数据。

1、recv()

原型:ssize_t recv(int sockfd, void *buf, size_t nbytes, int flags);

返回值:返回数据的字节长度;若无可用数据或对等方已经按序结束,返回0;若出错,返回-1.(APUE说法)

对于SOCK_STREAM套接字来讲,recv接收的数据可以比预期的少,recv的第四个参数可以选MSG_WAITALL标志来阻止这种行为,当flags为MSG_WAITALL时,recv会阻塞直到所指定的长度nbytes字节的数据全部返回,recv才会返回。

正常情况下recv 是会等待直到读取到nbytes长度的数据,但是这里的MSG_WAITALL也只是尽量读全,在有中断的情况下recv 还是可能会被打断,造成没有读完指定的nbytes的长度。使用这个标志recv会在以下三种情况发生时返回:1)当读到了指定的字节时,函数正常返回.返回值等于nbytes; 2)当读到了文件的结尾时,函数正常返回.返回值小于nbytes(不知道对于SOCK_STREAM字节流也是这样); 3)当操作发生错误时,返回-1,且设置错误为相应的错误号(errno)。

recv一次能接收的字节数nbytes应该与socket接收缓冲区的大小有关,当使用的套接字为SOCK_STREAM类型时,不能保证一次recv就能读取sockfd发送的所有数据,因此需要重复调用直到它返回0,可以采用如下方法实现:

while((n = recv(sockfd, buf, nbytes, 0)) > 0)

write(STDOUT_FILENO, buf, n);

对于SOCK_DGRAM和SOCK_SEQPACKET套接字,MSG_WAITALL并不改变什么,因为这些基于报文的套接字类型一次读取就返回整个报文。

如果flags为0,则recv和read一样。

2、ssize_t read(inf fd, void *buf, size_t bnytes);

在阻塞的tcp socket上使用read读取的数据长度和recv一样会发生返回值比指定长度短的情况。引用《UNIX网络编程 卷一 套接字联网API》3.9中的说法: 
字节流套接口(如tcp套接口)上的read和write函数所表现的行为不同于通常的文件IO。字节流套接口上的读或写输入或输出的字节数可能比要求的数量少,但这不是错误状况,原因是内核中套接口的缓冲区可能已达到了极限。此时所需的是调用者再次调用read或write函数,以输入或输出剩余的字节。 

可以使用readn函数来实现循环读取以解决这个问题:

ssize_t      /* Read "n" bytes from a descriptor. */
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;  /* and call read() again */
   else
    return(-1);
  } else if (nread == 0)
   break;    /* EOF */

  nleft -= nread;
  ptr   += nread;
 }
 return(n - nleft);  /* return >= 0 */
}

以上参考APUE和http://naso.iteye.com/blog/1927483http://blog.csdn.net/xjbclz/article/details/51834728等博文完成。

猜你喜欢

转载自blog.csdn.net/hhhlizhao/article/details/73912578