1 面向无连接的协议
先用socket( )或WSASocket()建立套接字,再把新创建的套接字和网络接口bind( )。和面向连接的套接字不同的是,我们不必调用listen( )和accept( ),直接发送或接收数据:
1.1 接收端
intrecvfrom( SOCKET s, char* buf, int len, int flags,
structsockaddr* from, int* fromlen );
intWSARecvFrom( SOCKET s, LPWSABUF lpBuffers, DWORD dwBufferCount,
LPDWORDlpNumberOfBytesRecvd, LPDWORD lpFlags,
structsockaddr* lpFrom, LPINT lpFromlen,
LPWSAOVERLAPPEDlpOverlapped,
LPWSAOVERLAPPED_COMPLETION_ROUTINElpCompletionRoutine
);
1.2 发送端
intsendto( SOCKET s, const char* buf, int len, int flags,
conststruct sockaddr* to, int tolen );
intWSASendTo( SOCKET s, LPWSABUF lpBuffers, DWORD dwBufferCount,
LPDWORDlpNumberOfBytesSent, DWORD dwFlags,
conststruct sockaddr* lpTo, int iToLen,
LPWSAOVERLAPPEDlpOverlapped,
LPWSAOVERLAPPED_COMPLETION_ROUTINElpCompletionRoutine
);
1.3 面向无连接的套接字的特点
面向无连接的协议几乎都是基于消息的协议,因此面向无连接的套接字也可以称为基于消息的套接字。需要注意以下几点:
在发送端,由于面向消息的协议对数据边界有保护,所以提交给发送函数的数据如果未能完全发送,发送函数就会返回错误。
在接收端,接收函数必须提供一个足够大的缓冲空间。如果提供的缓冲不够接收整个消息,接收函数就会返回错误。发生这种情况时,缓冲区会尽力接收数据,但未收完的数据会被丢弃。当然,如果使用的协议支持部分消息,就可用MSG_PARTIAL标志来避免数据的丢失。
综上所述,对于面向无连接的套接字来说,一次发送对应一次接收,发送和接收之间具有一一对应的关系。
注意:UDP套接字并不真正和网络接口绑定在一起。而是建立一种联系,即绑定的IP接口成为发送UDP数据报的源IP地址。路由表才真正决定数据报在哪个物理接口上发送出去。如果不调用bind( ),而是先调用sendto()或WSASendTo( )进行连接,网络堆栈就会根据路由表自动选出最佳IP地址,如果之前先执行了bind( ),源IP地址就会有误。
1.4 释放套接字资源
因为面向无连接的协议没有连接,所以也不会有正式的关闭。在接收端或发送端结束收发数据时,它只是在套接字上调用closesocket( ),便释放了为套接字分配的相关资源。