int sendmsg(int s,const struct msghdr * msg,int flags);
并且msghdr结构具有以下形式:
struct msghdr { void * msg_name; /* optional address */ socklen_t msg_namelen; /* size of address */ struct iovec * msg_iov; /* scatter/gather array */ size_t msg_iovlen; /* # elements in msg_iov */ void * msg_control; /* ancillary data,see below */ socklen_t msg_controllen; /* ancillary data buffer len */ int msg_flags; /* flags on received message */ };
如你所见,msghdr有一个缓冲区数组,iovec和缓冲区数msg_iovlen.我不知道sendmsg如何发送这些缓冲区.它连接所有缓冲区并发送还是发送在for循环?
解决方法
For
send()
andsendto()
,the message is found inbuf
and has length
len
. Forsendmsg()
,the message is pointed to by the elements of the
arraymsg.msg_iov
. Thesendmsg()
call also allows sending ancillary
data (also known as control information).
对于一个流套接字,无论如何.您发送的任何数据只会在另一端最后成为一长串数据.
对于数据报或消息套接字,我可以看到为什么更清晰一些将会有所帮助.但是,您似乎只发送一个数据报或消息与一个sndmsg调用;每个缓冲区元素不是一个.
我真的很好奇地挖掘了Linux源代码,并更好地了解了这个答案.看起来像send,sendto只是在Linux中的sendmsg的包装器,为你构建了struct msghdr.实际上,UDP sendmsg实现为每个sendmsg调用一个UDP头提供了空间.
如果性能是你担心的,那么如果你只传递一个iovec,那么看起来你不会受益于sendmsg.如果您在用户空间中连接缓冲区,那么这可能会赢得您的一些.
这与writev有点类似,附加的好处是您可以指定与无连接套接字(如UDP)一起使用的目标地址.你也可以添加辅助数据,如果你是这样的事情. (通常用于跨UNIX域套接字发送文件描述符.)