accept阻塞等待,close(fd)无作用

bind绑定端口之后,accept还在阻塞等待连接,这时如果另一个线程close(fd),此时的fd并没有解绑,该端口还是处于监听状态;导致其他线程绑定该端口时失败;要解决该问题,需要将accept函数改为非阻塞等待,并通过一个标志位来判断是否继续accept;

                while (1)
                {
                    FD_SET(p_session->sockfd, &rset);
                    ret_val = select(p_session->sockfd + 1, &rset, NULL, NULL, &accept_timeout);
                    if (ret_val > 0 && FD_ISSET(p_session->sockfd, &rset))
                    {
                        p_session->connfd = socket_accept(p_session->sockfd, &p_session->addr_client);
                        if (p_session->connfd < 0)
                        {
                            PRT_ERR("socket_accept faield, try again...\n");
                            continue;
                        }
                        else
                        {
                            PRT_DBG("socket_accept success, keepalive:%s\n", p_session->susp_param.keepalive);
                            break; // accept成功退出while(1)循环
                        }
                    }
                    if (!p_session->b_on_flying)
                    {
                        return;
                    }
                }

 

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
ftp客户端ftpclient纯C语言winsock实现socket编程 /* *本程序是2009年计算机网络课程设计作品。 *本程序参考rfc959标准。能准确与遵守此标准的服务器进行信息交互。 *本人不保留任何版权。 *本程序仅供学习研究测试使用。因使用本程序所有或部分代码所产生的任何后果,本人均不负任何法律责任。 *2009年7月13日 */ #include #include #include /*system()*/ #include #pragma comment(lib,"ws2_32.lib") #define ONUM 512 #define MNUM 512 #define FNUM 512 #define pt struct host { char ip[20]; unsigned short port; }; SOCKET ts; fd_set readfds; struct timeval timeval; struct host host; char renum[4]; char ordertemp[ONUM]; char order[ONUM]; char ordercp[ONUM]; char mess[MNUM]; char file[FNUM]; char setpath[FNUM]; char setpathf[FNUM]; int i,door,r,sys,seti;/*i for;door switch;r receive num;sys system state;set set state*/ char *p;/*strtok*/ int printmess(); void input(char ordertemp[]); int ftp(); int user(); int pass(); int command(); int list(); SOCKET createDataSocket(); int set(); int retr(); int stor(); int stor() { char filename[256]; SOCKET ds; int wi; int r2,r3,r4,bsnum,brnum; FILE *fp=NULL; set(); memset(filename,'\0',256); memset(order,'\0',ONUM); for(i=5;ibsnum); }/*while*/ printf("\n"); switch(printmess()) { case 425: case 426: case 451: case 551: case 552:closesocket(ds);fclose(fp);return -1; case 250: case 226:fclose(fp);return 0; default:return 0; } }/*stor*/ int retr() { char filename[256]; unsigned long fsize,wfsize; int r2,wi; SOCKET ds; FILE *fp=NULL; memset(filename,'\0',256); memset(order,'\0',ONUM); for(i=5;i<=200&&ordercp[i]!='\0';i++)order[i-5]=ordercp[i]; strcpy(filename,setpathf); strcat(filename,"\\\\"); strcat(filename,order); ds=createDataSocket(); if(ds==-1)return -1; memset(order,'\0',ONUM); strcpy(order,"type i\r\n");/*type i 二进制 type a ASCII*/ send(ts,order,strlen(order),0); switch(printmess()) { case 421:closesocket(ds);closesocket(ts);sys=0;return 421; case 530:closesocket(ds);sys=1;return -1; case 500: case 501: case 504: case 226:closesocket(ds);return -1; case 200:break; default:closesocket(ds);return -1; } strcat(ordercp,"\r\n"); send(ts,ordercp,strlen(ordercp),0); switch(printmess()) { case 421:closesocket(ds);closesocket(ts);sys=0;return 421; case 530:closesocket(ds);sys=1;return -1; case 450: case 500: case 501: case 550:closesocket(ds);return -1; case 125: case 150:break; default:closesocket(ds);return -1; } set(); system(setpath); fp=fopen(filename,"wb"); if(!fp){printf("create file fail!\n");closesocket(ds);printmess();return -1;} wi=1;fsize=0;wfsize=0; while(wi) { memset(file,'\0',FNUM); r=recv(ds,file,FNUM,0); if(r==SOCKET_ERROR) { printf("file recv error!\n"); closesocket(ds); fclose(fp); return -1; } fsize=fsize+r; if(r==0){wi=0;break;} printf("receive %8d bytes! have received %16d bytes!\r",r,fsize); r2=fwrite(file,sizeof(char),r,fp); fflush(fp);/*这里是关键*/ wfsize=wfsize+r2; printf("write %8d bytes!have written %20d bytes!\r",r2,wfsize); }/*while*/ printf("\n"); switch(printmess()) { case 425: case 426: case 451:closesocket(ds);fclose(fp);return -1; case 250: case 226: closesocket(ds); wi=1; while(wi)if(fclose(fp)==0)wi=0;return 0; default:return 0; } }/*retr()*/ 以下代码请下载本程序。VC++6.0编译通过
下面是一个使用 select 等待 accept 的示例代码: ```python import socket server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM) server_socket.bind(('localhost', 8000)) server_socket.listen(5) sockets_list = [server_socket] while True: read_sockets, _, _ = select.select(sockets_list, [], []) for sock in read_sockets: if sock == server_socket: client_socket, client_address = server_socket.accept() sockets_list.append(client_socket) print('Accepted new connection from {}:{}'.format(*client_address)) else: data = sock.recv(1024) if data: print(data.decode()) else: sock.close() sockets_list.remove(sock) print('Connection closed') ``` 在上面的代码中,我们首先创建了一个服务器套接字 `server_socket`,并将其绑定到本地主机的端口 8000 上。然后,我们使用 `select` 函数来等待新的连接。在每次循环中,我们调用 `select` 函数并传入一个包含 `server_socket` 的套接字列表 `sockets_list`,以及空的读取、写入和错误列表。`select` 函数将阻塞,直到有一个套接字处于可读状态,然后返回可读套接字的列表 `read_sockets`。 我们遍历 `read_sockets` 列表,如果其中的套接字是 `server_socket`,则表示有新的客户端连接请求。我们调用 `accept` 方法来接受连接请求,并将新的客户端套接字 `client_socket` 添加到 `sockets_list` 列表中。我们还打印出客户端的地址和端口号。 如果读取的套接字不是 `server_socket`,则表示有数据到达。我们使用 `recv` 方法来接收数据,并将其打印到控制台上。 最后,如果 `recv` 方法返回空数据,则表示客户端已关闭连接。我们关闭套接字并从 `sockets_list` 列表中删除该套接字。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值