1.怎样用setsockopt重新设置SOCKET缓冲区的大小
// 这段代码是改变接系统接收缓冲区大小。每次可以从另一端接收数据大小变大。
int nBufLen;
int nOptlLen;
nErrCode = getsockopt(s, SOL_SOCKET, SO_RCVBUF, (char*)nBufLen, &nOptlLen);
if (SOCKET_ERROR == nErrCode)
{
return EXIT_FAILURE;
}
nBufLen *= 10;
nErrCode = setsockopt(s, SOL_SOCKET, SO_RCVBUF, (char*)&nBufLen, nOptlLen);
if (SOCKET_ERROR == nErrCode)
{
return EXIT_FAILURE;
}
// 检查设置系统接收数据缓冲区是否成功
int uiNewRcvBuf;
int nOptLen;
getsockopt(s, SOL_SOCKET, SO_RCVBUF, (char*)uiNewRcvBuf, &nOptLen);
if (SOCKET_ERROR == nErrCode || uiNewRcvBuf != nBufLen)
{
return EXIT_FAILURE;
}
// 这一段,是更改发送缓冲区大小,使我们每次可以发送比较大的数据。
unsigned int uiOldSize = 0;
unsigned int uiNewSize = 0;
int uiRcvBufLen = 0;
if((uiOldSize = GetSysBuffSize(inSocket)) == 0)
{
// 获取缓冲大小失败
return false;
}
uiRcvBufLen = sizeof(uiBuffSize);
if (SOCKET_ERROR == setsockopt(inSocket, SOL_SOCKET, SO_SNDBUF, (char*)&uiBuffSize, uiRcvBufLen))
{
// 修改系统缓冲区大小失败
return false;
}
// 检查设置系统发送缓冲区是否成功
uiRcvBufLen = sizeof(uiNewSize);
if (SOCKET_ERROR == getsockopt(inSocket, SOL_SOCKET, SO_SNDBUF, (char*) &uiNewSize, &uiRcvBufLen) || uiNewSize == uiOldSize)
{
// 修改系统发送缓冲区失败
return false;
}
2.怎样用setsockopt重新设置SOCKET缓冲区的大小
// 这段代码是改变接系统接收缓冲区大小。每次可以从另一端接收数据大小变大。
int nBufLen;
int nOptlLen;
nErrCode = getsockopt(s, SOL_SOCKET, SO_RCVBUF, (char*)nBufLen, &nOptlLen);
if (SOCKET_ERROR == nErrCode)
{
return EXIT_FAILURE;
}
nBufLen *= 10;
nErrCode = setsockopt(s, SOL_SOCKET, SO_RCVBUF, (char*)&nBufLen, nOptlLen);
if (SOCKET_ERROR == nErrCode)
{
return EXIT_FAILURE;
}
// 检查设置系统接收数据缓冲区是否成功
int uiNewRcvBuf;
int nOptLen;
getsockopt(s, SOL_SOCKET, SO_RCVBUF, (char*)uiNewRcvBuf, &nOptLen);
if (SOCKET_ERROR == nErrCode || uiNewRcvBuf != nBufLen)
{
return EXIT_FAILURE;
}
// 这一段,是更改发送缓冲区大小,使我们每次可以发送比较大的数据。
unsigned int uiOldSize = 0;
unsigned int uiNewSize = 0;
int uiRcvBufLen = 0;
if((uiOldSize = GetSysBuffSize(inSocket)) == 0)
{
// 获取缓冲大小失败
return false;
}
uiRcvBufLen = sizeof(uiBuffSize);
if (SOCKET_ERROR == setsockopt(inSocket, SOL_SOCKET, SO_SNDBUF, (char*)&uiBuffSize, uiRcvBufLen))
{
// 修改系统缓冲区大小失败
return false;
}
// 检查设置系统发送缓冲区是否成功
uiRcvBufLen = sizeof(uiNewSize);
if (SOCKET_ERROR == getsockopt(inSocket, SOL_SOCKET, SO_SNDBUF, (char*) &uiNewSize, &uiRcvBufLen) || uiNewSize == uiOldSize)
{
// 修改系统发送缓冲区失败
return false;
}
3.SOCKET 缓冲区
按照unix/linux下面socket send的描述,对于非阻塞的socket,如果send时候buffer不足,send会失败 (返回-1)。
按照规矩,windows下的socket接口应该和unix/linux下面的兼容(TCP/IP协议栈就是unix下面开发出来的,而linux下的TCP/IP协议栈则是最标准的可看见源码参考物),所以你的情况应该也是返回失败。另外你说的锁住是对阻塞的socket而言,也就是如果是阻塞的socket,在没有空间的时候,send会被block住,直到socket的缓冲有地方。
4.如何获取SOCKET 发送缓冲区使用大小
解决思路:发送数据存在缓冲区中,我们判断发送缓冲区大小变化,即可获知发送是否成功;具体方法如下:发送数据后,获得已使用缓冲区大小buf,如果buf==0,表示成功,否则,表示未发送;
那么,如何来获得当前已占用发送缓冲区大小?
1. 第一步我们自然想到是否存在这样的sockopt接口
getsockopt(clientSocket, SOL_SOCKET, SO_SNDBUF, (void*)&sendbuflen, &len);
getsockopt中,有参数SO_SNDBUF,貌似是用来获得发送缓冲大小的;
但经过试验,我们发现 无论数据是否发送成功,该值一直不变;
查看内核代码,发现 该参数的含义是 总共的发送缓冲区,包括 已占用 和 空闲的;
2. 第二步,我们发现/proc/net/tcp 中tx_queue大小,即已占用发送缓冲区大小
/proc/net/tcp 中tx_queue 的计算:“tp->write_seq - tp->snd_una”,根据该计算方法,在内核源码中查找,发现内核导出了2个获得已使用发送缓冲的编程接口;
转载请注明出处windows之家 » win10如何设置socket缓冲区