首页 > 其他分享 >三-select模型

三-select模型

时间:2023-12-05 11:22:07浏览次数:44  
标签:函数 模型 fd 数组 接字 select socket

select模型是对简单C/S模型的优化,他解决了accept函数阻塞等待连接的问题。并且允许应用程序同时监视多个套接字,从而实现简单的并发请求。通过调用select函数确认一个或多个套接字当前的状态,并根据当前状态进行相应操作。在select模型模型中,select函数是最关键的。

select模型工作原理

select模型维护了一个Socket数组,通过遍历该数组检查当前是否存在就绪socket,并将所有就绪的socket返回。我们遍历该数组提供相应服务。工作原理大致如下:

  1. 将服务端Socket添加至socket数组中。
  2. 调用select()函数遍历socket数组。
  3. 返回就绪socket数组,对该数组集中处理。
  4. 如果是服务端Socket,调用accept()函数接收连接请求,并将该客户端数组添加至socket数组。
  5. 如果是客户端Socket,调用send()、recv()函数进行通信,当客户端下线时,从socket数组中移除该Socket。
  6. 重复执行2-6步。

select模型使用步骤

  1. 启动Socket服务
  2. 创建套接字
  3. 为套接字绑定端口信息
  4. 监听套接字
  5. 使用select模型监听套接字集合,处理数据
  6. 关闭套接字和网络服务

select()函数

该函数定义如下:

int select(int nfds, fd_set readfds, fd_set writefds, 
                fd_set exceptfds, const struct timeval timeout);

参数

  • nfds:忽略,传入0。为了与Berkeley 套接字兼容。
  • readfds:可读取套接字的集合,调用函数时传入要监视的套接字,函数返回时保存可读套接字。
  • writefds:可写套接字的集合。只要建立连接,则任何时候都可写入,可以传NULL。
  • exceptfds:异常套接字集合。
  • timeout:指定超时时间。

如果是服务端socket套接字,可读表示当前有客户端进行连接。如果是客户端套接字,可读表示当前有数据发送至服务端。select()函数是一个阻塞函数,如果指定了超时时间且没有socket就绪,select()函数返回。

返回值

  • 0:超时
  • >0:当前就绪的套接字数量,就绪的套接字保存在readfds中。
  • -1:出现错误,可以通过WSAGetLasterror()函数获取错误码。

fd_set结构体

fd_set结构体定义如下:

typedef struct fd_set {
        u_int   fd_count;               //fd_array数组中当前元素个数
        SOCKET  fd_array[FD_SETSIZE];   //socket数组
} fd_set;

在给select函数传递参数时,我们将需要监听的socket封装进fd_set结构体。select函数会顺序遍历数组,当发现有socket就绪,select函数返回,就绪的socket集合通常保存在readfds中。

下面列出了几个关于fd_set结构体的操作宏:

示例

  1 #define _WINSOCK_DEPRECATED_NO_WARNINGS
  2 
  3 #include <iostream>
  4 #include <WinSock2.h>
  5 #pragma comment(lib, "ws2_32.lib")
  6 using namespace std;
  7 const int nMajorVersion = 2;
  8 const int nMinorVersion = 2;
  9 
 10 int main()
 11 {
 12     DWORD dwVersion = MAKEWORD(nMajorVersion, nMinorVersion);
 13     WSADATA wsaData;
 14     int nStartRet = WSAStartup(dwVersion, &wsaData);
 15     if (nStartRet != 0)
 16     {
 17         cout << "WSAStartup failed with error :" << nStartRet << endl;
 18         return 1;
 19     }
 20 
 21     if (LOBYTE(wsaData.wVersion) != nMajorVersion || HIBYTE(wsaData.wVersion) != nMinorVersion)
 22     {
 23         cout << "version failed" << endl;
 24         WSACleanup();
 25         return 1;
 26     }
 27 
 28     SOCKET sockServer = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
 29     if (sockServer == INVALID_SOCKET)
 30     {
 31         cout << "socket create failed with code: "<< WSAGetLastError() << endl;
 32         WSACleanup();
 33         return 1;
 34     }
 35 
 36     sockaddr_in addInfo;
 37     addInfo.sin_family = AF_INET;
 38     addInfo.sin_port = htons(12345);
 39     addInfo.sin_addr.S_un.S_addr = inet_addr("127.0.0.1");
 40     int nBindRet = bind(sockServer, reinterpret_cast<sockaddr*>(&addInfo), sizeof(addInfo));
 41     if (SOCKET_ERROR == nBindRet)
 42     {
 43         cout << "bind failed with code: " << WSAGetLastError() << endl;
 44         closesocket(sockServer);
 45         WSACleanup();
 46         return 1;
 47     }
 48 
 49     int nListenRet = listen(sockServer, SOMAXCONN);
 50     if (SOCKET_ERROR == nBindRet)
 51     {
 52         cout << "bind failed with code: " << WSAGetLastError() << endl;
 53         closesocket(sockServer);
 54         WSACleanup();
 55         return 1;
 56     }
 57 
 58     fd_set allSockets;
 59     FD_ZERO(&allSockets);
 60     FD_SET(sockServer, &allSockets);
 61     struct timeval tv = { 3,0 };
 62 
 63     while (1)
 64     {
 65         fd_set tmpAllSockets = allSockets;
 66         int nSelectRet = select(0, &tmpAllSockets, NULL, NULL, &tv);
 67         if(nSelectRet == 0) continue;        //超时
 68         else if (nSelectRet == -1)            //出错
 69         {
 70             cout << "select failed with code: " << WSAGetLastError() << endl;
 71             break;
 72         }
 73         else if (nSelectRet > 0)
 74         {
 75             for (int i = 0; i < tmpAllSockets.fd_count; i++)
 76             {
 77                 SOCKET socketID = tmpAllSockets.fd_array[i];
 78 
 79                 if (socketID == sockServer)
 80                 {
 81                     SOCKET socketClient = accept(socketID, NULL, NULL);
 82                     if (socketClient == INVALID_SOCKET)
 83                         continue;
 84 
 85                     FD_SET(socketClient, &allSockets);
 86                 }
 87                 else
 88                 {
 89                     char buf[1024] = { 0 };
 90                     int nRecvRet = recv(socketID, buf, 1024, 0);
 91                     if (nRecvRet == 0)        //客户端断开连接
 92                     {
 93                         FD_CLR(socketID, &allSockets);
 94                         closesocket(socketID);
 95                         continue;
 96                     }
 97                     else if (nRecvRet > 0)
 98                     {
 99                         cout << socketID << " : " << buf << endl;
100                         send(socketID, "我收到了你发送的数据!", sizeof("我收到了你发送的数据!"), 0);
101                     }
102                     else if (nRecvRet == SOCKET_ERROR)
103                     {
104                         int nError = WSAGetLastError();
105                         if (nError == 10054)
106                         {
107                             FD_CLR(socketID, &allSockets);
108                             closesocket(socketID);
109                             continue;
110                         }
111                         else
112                             cout << "recv error." << endl;
113                     }
114                 }
115             }
116         }
117     }
118 
119     for (int i = 0; i < allSockets.fd_count; i++)
120     {
121         closesocket(allSockets.fd_array[i]);
122     }
123     FD_ZERO(&allSockets);
124     WSACleanup();
125     return 0;
126 }

总结

select模型有以下优点:

  • select允许同时监视多个套接字的读写状态,使得在单个线程中可以处理多个套接字的I/O操作,提高了系统的效率。
  • select模型使用简单,适合初学者设计简单的网络通信。

select模型有以下缺点:

  • select()函数本身会造成阻塞。
  • 由于需要遍历整个被监视的套接字集合,在大规模并发场景下,select模型性能低下。
  • 受限于FD_SET集合有大小限制,后续的拓展性差,无法适用于高并发场景。
  • 无法处理大量数据,如果有大量数据需要处理,会导致阻塞其他套接字的读写。

标签:函数,模型,fd,数组,接字,select,socket
From: https://www.cnblogs.com/BroccoliFighter/p/17876823.html

相关文章

  • HTTPS处于OSI模型中哪个层?
    HTTPS(HypertextTransferProtocolSecure)位于OSI(OpenSystemsInterconnection)模型的应用层和传输层之间。在OSI模型中,从底层到顶层依次是:物理层、数据链路层、网络层、传输层、会话层、表示层和应用层。HTTPS是在应用层上运行的协议,它基于HTTP并添加了安全性的扩展,通过使......
  • 关于三维模型几何坐标校正的技术方法探讨
    关于三维模型几何坐标校正的技术方法探讨 三维模型的几何坐标校正是确保模型精度和一致性的关键步骤。下面将分析几个主要的技术方法用于倾斜摄影三维模型几何坐标校正。1、地面控制点校正:使用地面控制点进行几何坐标校正是最常见和直接的方法。首先,在倾斜摄影数据中确定地面......
  • 因果推断9-18 链状结构、叉状结构、对撞结构、D-分割、模型检验和等价类
     https://www.bilibili.com/video/BV1tk4y127L1/?spm_id_from=333.788&vd_source=3ad05e655a5ea14063a9fd1c0dcdee3e所以得到一个结论,如果在一个链结构里面,比如X->Y->Z,condition到中间变量Y时,那么两边的变量X和Z是独立的。如果不condition到中间的变量时,那么两边的变量X,Z是......
  • 使用 PyTorch 完全分片数据并行技术加速大模型训练
    本文,我们将了解如何基于PyTorch最新的完全分片数据并行(FullyShardedDataParallel,FSDP)功能用Accelerate库来训练大模型。动机......
  • Day12 jvm 内存模型JMM
    1.jvm内存模型JMM原帖链接JMM控制Java线程之间的通信,决定一个线程对共享变量的写入何时对另一个线程可见。每条线程在自己的工作内存中对共享变量(副本)进行操作,JMM再负责把这些操作同步到主内存中JVM1.8用Metaspace(元空间)(在JVM外的本地内存中)取代了方法区(MethodArea)(在......
  • 做算力的浪潮信息为什么还要再卷大模型?
    避免重复造轮子,前提是轮子已经造得很好。 大模型有多卷?现在国内已经有180个以上生成式大模型,科技大厂、互联网大厂纷纷入局,既有百度、浪潮信息、阿里、腾讯等一众巨头,也有专攻AI的讯飞、商汤等垂直领域小巨头,以及“日日新”的创业企业。今天A厂商发布大模型,各种参数对比下来......
  • 从HumanEval到CoderEval: 你的代码生成模型真的work吗?
    本文分享自华为云社区《从HumanEval到CoderEval:你的代码生成模型真的work吗?》,作者:华为云PaaS服务小智。本文主要介绍了一个名为CoderEval的代码生成大模型评估基准,并对三个代码生成模型(CodeGen、PanGu-Coder和ChatGPT)在该基准上的表现进行了评估和比较。研究人员从真实的开源......
  • LangChain调用本地模型
    学习LangChain参考https://python.langchain.com.cn/docs/get_started/quickstart调用本地下载的模型参考https://blog.csdn.net/qq_43692950/article/details/131743987在JupyterNotebook中试验的代码(注意Jupyter不会释放GPU显存)fromlangchainimportPromptTemplate,LL......
  • 从Hugging Face下载模型到本地并调用
    不同的模型需要的显存不同,下载前先查一下自己GPU能支持什么模型 1.用如下脚本可以下载HuggingFace上的各种模型,网址 https://huggingface.co/modelsdownload.py#coding=gbkimporttimefromhuggingface_hubimportsnapshot_download#huggingface上的模型名称repo_id......
  • 他山之石,可以攻玉|银行业数据中心数字化转型之模型篇 04(完结)
    ​导语: 银行业数据中心数字化转型是一项系统性工程既涉及管理层面转型——包括数字化转型战略、基础架构和技术架构转型、技术创新和知识体系转型,又涉及执行层面转型——包括人员管理(P)、流程管理(P)、技术管理(T)、资源管理(R)等。数据中心数字化转型作为一项宏大的系统性工程,必须要......