前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >ioctlsocket() 用法 socket recvfrom 阻塞 非阻塞 设置

ioctlsocket() 用法 socket recvfrom 阻塞 非阻塞 设置

作者头像
战神伽罗
发布2019-08-29 11:29:56
3.4K0
发布2019-08-29 11:29:56
举报

参考一

Select模型与ioctlsocket的使用方法

int iMode = 1; //0:阻塞 ioctlsocket(socketc,FIONBIO, (u_long FAR*) &iMode);//非阻塞设置 rs=recvfrom(socketc,rbuf,sizeof(rbuf),0,(SOCKADDR*)&addr,&len); int ioctlsocket (SOCKET s, long cmd, u_long FAR* argp ); s [in] A descriptor identifying a socket. cmd [in] The command to perform on the socket s. argp [in/out] A pointer to a parameter for cmd.

不知道大家有没有遇到过这种情况,当socket进行TCP连接的时候(也就是调用connect时),一旦网络不通,或者是ip地址无效,就可能使整个线程阻塞。一般为30秒(我测的是20秒)。如果设置为非阻塞模式,能很好的解决这个问题,我们可以这样来设置非阻塞模式:

调用ioctlsocket函数:

unsigned long flag=1; if (ioctlsocket(sock,FIONBIO,&flag)!=0) { closesocket(sock); return false; } 以下是对ioctlsocket函数的相关解释: int PASCAL FAR ioctlsocket( SOCKET s, long cmd, u_long FAR* argp); s:一个标识套接口的描述字。 cmd:对套接口s的操作命令。 argp:指向cmd命令所带参数的指针。

注释: 本函数可用于任一状态的任一套接口。它用于获取与套接口相关的操作参数,而与具体协议或通讯子系统无关。支持下列命令: FIONBIO:允许或禁止套接口s的非阻塞模式。argp指向一个无符号长整型。如允许非阻塞模式则非零,如禁止非阻塞模式则为零。当创建一个套接口时,它就处于阻塞模式(也就是说非阻塞模式被禁止)。这与BSD套接口是一致的。WSAAsynSelect()函数将套接口自动设置为非阻塞模式。如果已对一个套接口进行了WSAAsynSelect() 操作,则任何用ioctlsocket()来把套接口重新设置成阻塞模式的试图将以WSAEINVAL失败。为了把套接口重新设置成阻塞模式,应用程序必须首先用WSAAsynSelect()调用(IEvent参数置为0)来禁至WSAAsynSelect()。 FIONREAD:确定套接口s自动读入的数据量。argp指向一个无符号长整型,其中存有ioctlsocket()的返回值。如果s是SOCKET_STREAM类型,则FIONREAD返回在一次recv()中所接收的所有数据量。这通常与套接口中排队的数据总量相同。如果S是SOCK_DGRAM 型,则FIONREAD返回套接口上排队的第一个数据报大小。 SIOCATMARK:确实是否所有的带外数据都已被读入。这个命令仅适用于SOCK_STREAM类型的套接口,且该套接口已被设置为可以在线接收带外数据(SO_OOBINLINE)。如无带外数据等待读入,则该操作返回TRUE真。否则的话返回FALSE假,下一个recv()或recvfrom()操作将检索“标记”前一些或所有数据。应用程序可用SIOCATMARK操作来确定是否有数据剩下。如果在“紧急”(带外)数据前有常规数据,则按序接收这些数据(请注意,recv()和recvfrom()操作不会在一次调用中混淆常规数据与带外数据)。argp指向一个BOOL型数,ioctlsocket()在其中存入返回值。 此时已经设置非阻塞模式,但是并没有设置connect的连接时间,我们可以通过调用select语句来实现这个功能。以下代码设定了是连接时间为5秒,如果还未能连上,则直接返回。

struct timeval timeout ; fd_set r; int ret; connect( sock, (LPSOCKADDR)sockAddr, sockAddr.Size()); FD_ZERO(&r); FD_SET(sock,&r); timeout.tv_sec = 5; timeout.tv_usec =0; ret = select(0,0,&r,0,&timeout); if ( ret <= 0 ) { closesocket(sock); return false; }

以下是对select函数的解释: int select ( int nfds, fd_set FAR * readfds, fd_set FAR * writefds, fd_set FAR * exceptfds, const struct timeval FAR * timeout ); 第一个参数nfds沒有用,仅仅为与伯克利Socket兼容而提供。 readfds指定一個Socket数组(应该是一个,但这里主要是表现为一个Socket数组),select检查该数组中的所有Socket。如果成功返回,则readfds中存放的是符合‘可读性’条件的数组成员(如缓冲区中有可读的数据)。 writefds指定一个Socket数组,select检查该数组中的所有Socket。如果成功返回,则writefds中存放的是符合‘可写性’条件的数组成员(如连接成功)。 exceptfds指定一个Socket数组,select检查该数组中的所有Socket。如果成功返回,则cxceptfds中存放的是符合‘有异常’条件的数组成员(如连接接失败)。 timeout指定select执行的最长时间,如果在timeout限定的时间内,readfds、writefds、exceptfds中指定的Socket沒有一个符合要求,就返回0。 如果对 Connect 进行非阻塞调用,则可读意味着已经成功连接,连接不成功则不可读。所以通过这样的设定,我们就能够实现对connect连接时间的修改。但是,应该注意,这样的设置并不能保证在限定时间内连接不上就说明网络不通。比如我们设的时间是5秒,但是由于种种原因,可能第6秒就能连接上,但是函数在5秒后就返回了。

参考二

ioctlsocket() 用法

◆I/O操作函数:主要用于获取与套接字相关的操作参数。 int ioctlsocket(SOCKET s, long cmd, u_long FAR * argp ); s为I/O操作的套接字。 cmd为对套接字的操作命令。 argp为命令所带参数的指针。 常见的命令: //确定套接字自动读入的数据量 #define FIONREAD _IOR(''''f'''', 127, u_long) /* get # bytes to read */ //允许或禁止套接字的非阻塞模式,允许为非0,禁止为0 #define FIONBIO _IOW(''''f'''', 126, u_long) /* set/clear non-blocking i/o */ //确定是否所有带外数据都已被读入 #define SIOCATMARK _IOR(''''s'''', 7, u_long) /* at oob mark? */ 本函数可用于任一状态的任一套接口。它用于获取与套接口相关的操作参数, 而与具体协议或通讯子系统无关。支持下列命令:

FIONBIO:允许或禁止套接口s的非阻塞模式。argp指向一个无符号长整型。如 允许非阻塞模式则非零,如禁止非阻塞模式则为零。当创建一个套接口时,它就 处于阻塞模式(也就是说非阻塞模式被禁止)。这与BSD套接口是一致的。WSAAs ynSelect()函数将套接口自动设置为非阻塞模式。如果已对一个套接口进行了WS AAsynSelect() 操作,则任何用ioctlsocket()来把套接口重新设置成阻塞模式的 试图将以WSAEINVAL失败。为了把套接口重新设置成阻塞模式,应用程序必须首先 用WSAAsynSelect()调用(IEvent参数置为0)来禁至WSAAsynSelect()。

FIONREAD:确定套接口s自动读入的数据量。argp指向一个无符号长整型,其中 存有ioctlsocket()的返回值。如果s是SOCKET_STREAM类型,则FIONREAD返回在一 次recv()中所接收的所有数据量。这通常与套接口中排队的数据总量相同。如果 S是SOCK_DGRAM 型,则FIONREAD返回套接口上排队的第一个数据报大小。

SIOCATMARK:确实是否所有的带外数据都已被读入。这个命令仅适用于SOCK_S TREAM类型的套接口,且该套接口已被设置为可以在线接收带外数据(SO_OOBINL INE)。如无带外数据等待读入,则该操作返回TRUE真。否则的话返回FALSE假, 下一个recv()或recvfrom()操作将检索“标记”前一些或所有数据。应用程序可 用SIOCATMARK操作来确定是否有数据剩下。如果在“紧急”(带外)数据前有常 规数据,则按序接收这些数据(请注意,recv()和recvfrom()操作不会在一次调 用中混淆常规数据与带外数据)。argp指向一个BOOL型数,ioctlsocket()在其中 存入返回值。 兼容性: 本函数为Berkeley套接口函数ioctl()的一个子集。其中没有与FIOASYNC等价的 命令,SIOCATMARK是套接口层次支持的唯一命令。 返回值: 成功后,ioctlsocket()返回0。否则的话,返回SOCKET_ERROR错误,应用程序 可通过WSAGetLastError()获取相应错误代码。 错误代码: WSANOTINITIALISED:在使用此API之前应首先成功地调用WSAStartup()。 WSAENETDOWN:WINDOWS套接口实现检测到网络子系统失效。 WSAEINVAL:cmd为非法命令,或者argp所指参数不适用于该cmd命令,或者该命 令 不适用于此种类型的套接口。 WSAEINPROGRESS:一个阻塞的WINDOWS套接口调用正在运行中。 WSAENOTSOCK:描述字不是一个套接口。

参考三

socket设置 ioctlsocket()

 简述:   控制套接口的模式。   #include <winsock.h>   int PASCAL FAR ioctlsocket( SOCKET s, long cmd, u_long FAR* argp);   s:一个标识套接口的描述字。   cmd:对套接口s的操作命令。   argp:指向cmd命令所带参数的指针。   注释:   本函数可用于任一状态的任一套接口。它用于获取与套接口相关的操作参数,而与具体协议或通讯子系统无关。支持下列命令: FIONBIO:允许或禁止套接口s的非阻塞模式。argp指向一个无符号长整型。如允许非阻塞模式则非零,如禁止非阻塞模式则为零。 当创建一个套接口时,它就处于阻塞模式(也就是说非阻塞模式被禁止)。这与BSD套接口是一致的。WSAAsynSelect()函数将套接口自动设置为非阻塞模式。 如果已对一个套接口进行了WSAAsynSelect() 操作,则任何用ioctlsocket()来把套接口重新设置成阻塞模式的试图将以WSAEINVAL失败。 为了把套接口重新设置成阻塞模式,应用程序必须首先用WSAAsynSelect()调用(IEvent参数置为0)来禁至WSAAsynSelect()。 FIONREAD:确定套接口s自动读入的数据量。argp指向一个无符号长整型,其中存有ioctlsocket()的返回值。如果s是SOCKET_STREAM类型, 则FIONREAD返回在一次recv()中所接收的所有数据量。这通常与套接口中排队的数据总量相同。如果S是SOCK_DGRAM 型, 则FIONREAD返回套接口上排队的第一个数据报大小。 SIOCATMARK:确实是否所有的带外数据都已被读入。这个命令仅适用于SOCK_STREAM类型的套接口,且该套接口已被设置为可以在线接收带外数据(SO_OOBINLINE)。 如无带外数据等待读入,则该操作返回TRUE真。否则的话返回FALSE假,下一个recv()或recvfrom()操作将检索“标记”前一些或所有数据。 应用程序可用SIOCATMARK操作来确定是否有数据剩下。如果在“紧急”(带外)数据前有常规数据,则按序接收这些数据(请注意,recv()和recvfrom()操作不会 在一次调用中混淆常规数据与带外数据)。argp指向一个BOOL型数,ioctlsocket()在其中存入返回值。 兼容性:   本函数为Berkeley套接口函数ioctl()的一个子集。其中没有与FIOASYNC等价的命令,SIOCATMARK是套接口层次支持的唯一命令。 返回值:   成功后,ioctlsocket()返回0。否则的话,返回SOCKET_ERROR错误,应用程序可通过WSAGetLastError()获取相应错误代码。 错误代码:   WSANOTINITIALISED:在使用此API之前应首先成功地调用WSAStartup()。   WSAENETDOWN:WINDOWS套接口实现检测到网络子系统失效。   WSAEINVAL:cmd为非法命令,或者argp所指参数不适用于该cmd命令,或者该命令   不适用于此种类型的套接口。   WSAEINPROGRESS:一个阻塞的WINDOWS套接口调用正在运行中。   WSAENOTSOCK:描述字不是一个套接口。

本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • Select模型与ioctlsocket的使用方法
  • ioctlsocket() 用法
  • socket设置 ioctlsocket()
相关产品与服务
命令行工具
腾讯云命令行工具 TCCLI 是管理腾讯云资源的统一工具。使用腾讯云命令行工具,您可以快速调用腾讯云 API 来管理您的腾讯云资源。此外,您还可以基于腾讯云的命令行工具来做自动化和脚本处理,以更多样的方式进行组合和重用。
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档