分享一下我老师大神的人工智能教程。零基础!通俗易懂!风趣幽默!还带黄段子!希望你也加入到我们人工智能的队伍中来!https://blog.csdn.net/jiangjunshow

我们知道, recv是一个阻塞的函数,没有到数据的时候,会傻傻地等待数据。那么怎么让recv函数有点脾气, 不傻傻地等待呢?

服务器为(先跑起来):

#include <stdio.h>#include <winsock2.h> // winsock接口#pragma comment(lib, "ws2_32.lib") // winsock实现int main(){ WORD wVersionRequested;  // 双字节,winsock库的版本 WSADATA wsaData;         // winsock库版本的相关信息  wVersionRequested = MAKEWORD(1, 1); // 0x0101 即:257  // 加载winsock库并确定winsock版本,系统会把数据填入wsaData中 WSAStartup( wVersionRequested, &wsaData ); // AF_INET 表示采用TCP/IP协议族 // SOCK_STREAM 表示采用TCP协议 // 0是通常的默认情况 unsigned int sockSrv = socket(AF_INET, SOCK_STREAM, 0); SOCKADDR_IN addrSrv; addrSrv.sin_family = AF_INET; // TCP/IP协议族 addrSrv.sin_addr.S_un.S_addr = inet_addr("0.0.0.0");  addrSrv.sin_port = htons(8888); // socket对应的端口 // 将socket绑定到某个IP和端口(IP标识主机,端口标识通信进程) bind(sockSrv,(SOCKADDR*)&addrSrv, sizeof(SOCKADDR)); // 将socket设置为监听模式,5表示等待连接队列的最大长度 listen(sockSrv, 5); SOCKADDR_IN addrClient; int len = sizeof(SOCKADDR);  while(1) {  unsigned int sockConn = accept(sockSrv,(SOCKADDR*)&addrClient, &len);  printf("accept is returned\n");  Sleep(3000);  char sendBuf[100] = "hello";  send(sockConn, sendBuf, strlen(sendBuf) + 1, 0); // 发送数据到客户端,最后一个参数一般设置为0  //closesocket(sockConn);  } closesocket(sockSrv); WSACleanup();  return 0;}

我们看到,当建立一个TCP连接后,服务器需要等3秒才发送数据,那么客户等得及吗?客户有耐心吗?且看客户:

#include <winsock2.h>#include <stdio.h>#pragma comment(lib, "ws2_32.lib")int main(){ WORD wVersionRequested; WSADATA wsaData; wVersionRequested = MAKEWORD(1, 1);  WSAStartup( wVersionRequested, &wsaData ); SOCKET sockClient = socket(AF_INET, SOCK_STREAM, 0); SOCKADDR_IN addrSrv; addrSrv.sin_addr.S_un.S_addr = inet_addr("127.0.0.1"); addrSrv.sin_family = AF_INET; addrSrv.sin_port = htons(8888); int ret = connect(sockClient, (SOCKADDR*)&addrSrv, sizeof(SOCKADDR)); printf("connect ret is %d\n", ret);          char recvBuf[200] = "x"; ret = recv(sockClient, recvBuf, 100, 0); if(0 > ret) {  printf("recv error");  return -3; } if(0 == ret) {  printf("recv ret is %d\n", ret);  return -4; }  printf("%s\n", recvBuf); while(1); closesocket(sockClient); WSACleanup(); return 0;}

运行客户端程序,可以发现,尽管服务器故意延时了3秒,但客户有无限的耐心,仍然能收到数据。

下面,我们来看看耐心有限的客户:

#include <winsock2.h>#include <stdio.h>#pragma comment(lib, "ws2_32.lib")int main(){ WORD wVersionRequested; WSADATA wsaData; wVersionRequested = MAKEWORD(1, 1);  WSAStartup( wVersionRequested, &wsaData ); SOCKET sockClient = socket(AF_INET, SOCK_STREAM, 0); SOCKADDR_IN addrSrv; addrSrv.sin_addr.S_un.S_addr = inet_addr("127.0.0.1"); addrSrv.sin_family = AF_INET; addrSrv.sin_port = htons(8888); int ret = connect(sockClient, (SOCKADDR*)&addrSrv, sizeof(SOCKADDR)); printf("connect ret is %d\n", ret);  fd_set rfds; struct timeval timeout = {4, 0};    FD_ZERO(&rfds);     FD_SET(sockClient, &rfds);    ret=select(-1, &rfds, NULL, NULL, &timeout);    printf("select ret is %d\n",ret);    if(0 > ret)    {  printf("select error\n");        return -1;    }    if(0 == ret)    {        printf("time out\n");  return -2;    }        if(FD_ISSET(sockClient, &rfds))    {        char recvBuf[200] = "x";  ret = recv(sockClient, recvBuf, 100, 0);  if(0 > ret)  {   printf("recv error");   return -3;  }  if(0 == ret)  {   printf("recv ret is %d, buf is %s\n", ret, recvBuf);   return -4;  }    printf("%s\n", recvBuf);    } while(1); closesocket(sockClient); WSACleanup(); return 0;}

这个客户的耐心只有4秒,他发飙说:服务器,你妈的,在我跟你建立TCP关系后,你如果4秒之内没发数据,我就不鸟你了。运行程序,可以看到,这个客户依然可以接受到数据。

下面我们再看看, 耐心只有2秒的客户:

#include <winsock2.h>#include <stdio.h>#pragma comment(lib, "ws2_32.lib")int main(){ WORD wVersionRequested; WSADATA wsaData; wVersionRequested = MAKEWORD(1, 1);  WSAStartup( wVersionRequested, &wsaData ); SOCKET sockClient = socket(AF_INET, SOCK_STREAM, 0); SOCKADDR_IN addrSrv; addrSrv.sin_addr.S_un.S_addr = inet_addr("127.0.0.1"); addrSrv.sin_family = AF_INET; addrSrv.sin_port = htons(8888); int ret = connect(sockClient, (SOCKADDR*)&addrSrv, sizeof(SOCKADDR)); printf("connect ret is %d\n", ret);  fd_set rfds; struct timeval timeout = {2, 0};    FD_ZERO(&rfds);     FD_SET(sockClient, &rfds);    ret=select(-1, &rfds, NULL, NULL, &timeout);    printf("select ret is %d\n",ret);    if(0 > ret)    {  printf("select error\n");        return -1;    }    if(0 == ret)    {        printf("time out\n");  return -2;    }        if(FD_ISSET(sockClient, &rfds))    {        char recvBuf[200] = "x";  ret = recv(sockClient, recvBuf, 100, 0);  if(0 > ret)  {   printf("recv error");   return -3;  }  if(0 == ret)  {   printf("recv ret is %d, buf is %s\n", ret, recvBuf);   return -4;  }    printf("%s\n", recvBuf);    } while(1); closesocket(sockClient); WSACleanup(); return 0;}

这个客户的脾气够火爆啊,等了2秒,就不等了,所以就没有收到数据。

在实际的软件开发中,常常要用到超时机制,可以避免客户端死等。

多的不闲扯了。最后,非常非常有趣的是:如果改为

   char recvBuf[2] = "x";

那么上面的各个程序会产生非常非常奇妙的结果,自己亲自研究一下吧!

靠,很晚了,睡觉。

分享一下我老师大神的人工智能教程。零基础!通俗易懂!风趣幽默!还带黄段子!希望你也加入到我们人工智能的队伍中来!https://blog.csdn.net/jiangjunshow

如何让recv函数有点脾气 (让你知道select函数的威力)相关推荐

  1. linux select函数返回值,socket中的select函数使用

    socket select()函数用于定时检查以创建的socket句柄的状态,以确定设定的socket句柄当前是否可写或有数据可读. select函数提供了一种方法,使得程序在操作socket时(如r ...

  2. 计算机函数select,请问关于select函数的操作程序?

    使用select函数的部分代码如下: //设置超时时间 timeval *ptimeval = new timeval; ptimeval.tv_sec = 60; ptimeval.tv_usec ...

  3. C语言中select函数简介及使用

    select函数用来检查套接字描述符(sockets descriptors)是否已准备好读/写,提供了一种同时检查多个套接字的方法. Linux中select函数的声明在/usr/include/x ...

  4. C语言中select函数的使用

    一般用connect.accept.recv或recvfrom这类函数,程序阻塞,直至该套接字上接受到数据后程序才能继续运行.但是使用select函数可以实现非阻塞方式的程序.它能够监视我们需要监视的 ...

  5. 从select函数谈及系统调用原理

    目录 系统调用流程 select函数 在glibc-2.28中查找select函数 在linux内核linux-4.20.1中查找select 参考: linux的select机制在socket编程中 ...

  6. c语言select函数头文件,简述select函数的功能与调用方法

    使用select 首先我们需要解答一些疑惑.为什么要使用select函数? 比如你有一个服务器程序,维护着N多个TCP连接,你如何去判断这些TCP连接上有数据传送过来了呢?最傻瓜的办法就是隔一定时间去 ...

  7. socket通信时如何判断当前连接是否断开--select函数,心跳线程,QsocketNotifier监控socket...

    client与server建立socket连接之后,如果突然关闭server,此时,如果不在客户端close(socket_fd),会有不好的影响: QsocketNotifier监控socket的槽 ...

  8. select()函数以及FD_ZERO、FD_SET、FD_CLR、FD_ISSET(转)

    select函数用于在非阻塞中,当一个套接字或一组套接字有信号时通知你,系统提供select函数来实现多路复用输入/输出模型, 原型: int select(int maxfd,fd_set *rds ...

  9. linux 下 select 函数的用法

    一.Select 函数详细介绍 Select在Socket编程中还是比较重要的,可是对于初学Socket的人来说都不太爱用Select写程序,他们只是习惯写诸如connect. accept.recv ...

最新文章

  1. 千万级、百万级数据删除优化
  2. 用installshield打包的asp.net程序
  3. 【Java学习笔记七】常用数据对象之数组
  4. CSS height高度 和 width宽度
  5. FreeEIM 2.0 beta 发布
  6. 如何在M1的Mac上检查App是原生执行,还是Rosetta转译后执行?
  7. Apache Hadoop 2.4.1完全分布式集群安装
  8. centos6 yum安装tomcat8
  9. python zip压缩文件
  10. Java设置定时器一种简单方法
  11. LeetCode#860: 柠檬水找零
  12. DC Administration Services 宣布ISDA裁决委员会2020年申请流程
  13. 【编译原理】语言认知之Java、Python、C++快速排序三者运行效率与开发效率比较
  14. UNI-APP 开发微信公众号(H5)JSSDK的使用、微信扫一扫
  15. ARM920T内存管理单元MMU
  16. IP 地址和子网掩码
  17. subprocess.Popen()的用法
  18. 当年的你是如何走上编程之路的?
  19. HTML学习笔记——各种居中对齐
  20. JavaScript制作简易单词测试题库

热门文章

  1. 带通滤波器中心频率计算公式中R是哪个值_磁珠及其在开关电源中的应用
  2. nvue 怎么布局
  3. 移动端判断手指上滑下滑
  4. 四史教育自动答题思路讲解(附加Java代码)
  5. 30本pdf完整版的经典Linux学习和开发教程和资料下载 android arm java 资料大全...
  6. Antd Pro项目接入qiankun微前端
  7. 智慧营区管理系统软件:提供智慧营区一体化平台解决方案
  8. 西游记中三位隐居世外的高人
  9. 信息系统项目管理师---第七章项目成本管理历年考题
  10. 好用的视频消重 如何查看视频md5