socket非阻塞模式 java_ioctlsocket 非阻塞模式的设置、设置socket为非阻塞模式

int iMode = 1; //0:阻塞

ioctlsocket(socketc,FIONBIO, (u_long FAR*) &iMode);//非阻塞设置

rs=recvfrom(socketc,rbuf,sizeof(rbuf),0,(SOCKADDR*)&addr,&len);

int ioctlsocket (

SOCKETs,

longcmd,

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秒后就返回了。

在Windows中,可以使用ioctlsocket函数将socket设置非阻塞模式。方法是先使用socket函数创建套接字,然后通过调用ioctlsocket函数,将套接字设置非阻塞模式。具体的代码如下: unsigned long ul = 1; SOCKET s = socket(AF_INET, SOCK_STREAM, 0); int ret = ioctlsocket(s, FIONBIO, (unsigned long *)&ul); if(ret == SOCKET_ERROR) { // 设置失败的处理 } 上述代码中,通过将ul设置为1,将套接字s设置非阻塞模式。如果设置失败,可以在后续进行相应的处理。在Linux中,可以使用fcntl函数将socket设置非阻塞模式。具体的代码如下: int flags = fcntl(socket, F_GETFL, 0); fcntl(socket, F_SETFL, flags | O_NONBLOCK); 上述代码中,首先通过fcntl函数获取套接字的当前标志位,然后使用F_SETFL将标志位设置为flags | O_NONBLOCK,即将非阻塞标志位添加到原有标志位上,从而将套接字设置非阻塞模式。如果需要将非阻塞模式设置回阻塞模式,可以使用以下代码: int flags = fcntl(socket, F_GETFL, 0); fcntl(socket, F_SETFL, flags & ~O_NONBLOCK) 上述代码中,通过将flags与~O_NONBLOCK进行按位与操作,将非阻塞标志位从原有标志位中移除,从而将套接字设置回阻塞模式。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *3* [C++ 示例代码 socket设置非阻塞方式(windows和linux)](https://blog.csdn.net/u011109761/article/details/128671637)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT0_1"}}] [.reference_item style="max-width: 50%"] - *2* [socket的阻塞与非阻塞](https://blog.csdn.net/u014779536/article/details/115595535)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT0_1"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值