recv & recvfrom

#include <sys/types.h>
#include <sys/socket.h>

ssize_t recv(int sockfd, void *buf, size_t len, int flags);

ssize_t recvfrom(int sockfd, void *buf, size_t len, int flags, struct sockaddr *src_addr, socklen_t *addrlen);

recv(),recvfrom()调用被用于从套接字接收消息。 它们可用于在无连接和面向连接的套接字上接收数据。正如,recv()和read()之间的唯一区别是标志的存在,使用零标志参数时,recv()通常等效于read()。同理,recv(sockfd,buf,len,flags)等价于recvfrom(sockfd,buf,len,flags,NULL,NULL)。

成功完成后,这两个调用都将返回消息的长度。 如果消息太长而无法容纳在提供的缓冲区中,则多余的字节可能会被丢弃,此时,返回值则取决于接收消息的套接字类型。

如果套接字上没有可用的消息,则接收调用将等待消息到达,除非套接字是非阻塞的,在这种情况下,将返回值-1且外部变量errno设置为EAGAIN 或EWOULDBLOCK。 接收呼叫通常返回任何可用数据,最多至请求的数量,而不是等待收到请求的全部数量。

flags参数

  flags参数是通过对以下一个或多个值进行“或”运算形成。

  MSG_CMSG_CLOEXEC   仅仅recvmsg()函数使用。

  MSG_DONTWAIT       启用非阻塞操作。 这提供与设置O_NONBLOCK标志类似的行为(通过fcntl(2)F_SETFL操作),但不同之处在于MSG_DONTWAIT是每个通话选项,而O_NONBLOCK是设置打开文件描述,这将影响在调用进程中的所有线程以及保存了同一文件描述符的其他进程。

  MSG_ERRQUEUE      This flag specifies that queued errors should be received from the socket error queue.

  MSG_OOB                                  这个标志请求接收正常数据流中不会接收到的带外数据。 一些协议将加急数据放在普通数据队列的开头,因此这标志不能与此类协议一起使用。

         MSG_PEEK                                 这个标志使接收操作从接收队列的开头返回数据,而不会从队列中删除该数据。 因此,后续的接收呼叫将返回相同的数据。

  MSG_TRUNC        对于原始(AF_PACKET),Internet数据报,netlink和UNIX数据报套接字:返回数据包或数据报的实际长度,即使它比传递的缓冲区长。

   MSG_WAITALL        这个标志导致请求阻塞,直到满足完整请求为止。 但是,如果捕获到信号,发生错误或断开连接,或者下一个要接收的数据与返回的数据类型不同,则呼叫返回的数据仍可能比请求的少。 这个标志对数据报套接字无效。  

recvfrom()

  recvfrom()将收到的消息放入缓冲区buf。 调用者必须指定缓冲区的大小。如果src_addr不为NULL,并且底层协议提供消息的源地址,则该源地址位于src_addr指向的缓冲区中。 在这种情况下,addrlen是一个值结果参数( a value-result argument)。 在调用之前,应将其初始化为src_addr缓冲区的大小。 返回时,addrlen则为源地址的实际大小。 如果提供的缓冲区太小,返回的地址将被截断。 在这种情况下,addrlen将返回一个大于提供给调用的值。如果调用方对源地址不感兴趣,则src_addr和addrlen应该指定为NULL。

recv()    

  recv()调用通常仅在面向连接的套接字上使用,等效于调用:recvfrom(fd,buf,len,flags,NULL,NULL)。

返回值

  这些调用返回接收到的字节数,如果发生了错误则返回-1。 如果发生错误,则将errno设置为指示错误。当流套接字对等方执行有序关闭时,返回值将为0(传统的“文件结束”返回)。各个域中的数据报套接字允许零长度的数据报。 收到这样的数据报后,返回值为0。如果请求从流套接字接收的字节数为0,则也可能返回值0。

原文地址:https://www.cnblogs.com/iuyy/p/13395198.html