一、服务器

1.服务器分类

单循环服务器:只能处理一个客户端任务的服务器
并发服务器:可同时处理多个客户端任务的服务器

二、TCP并发服务器的构建

1.如何构建?     

(1)多进程(每一次创建都非常耗时耗空间,但是安全)

#include "head.h"
int init_tcp(const char *ip, unsigned short port)
{int sockfd = socket(AF_INET, SOCK_STREAM, 0);if (sockfd < 0){perror("socket fail");return 1;}struct sockaddr_in seraddr;seraddr.sin_family = AF_INET;seraddr.sin_port = htons(port);seraddr.sin_addr.s_addr = inet_addr(ip);int ret = bind(sockfd, (struct sockaddr *)&seraddr, sizeof(seraddr));if (ret < 0){perror("bind fail");return 1;}ret = listen(sockfd, 100);if (ret < 0){perror("lisen fail");return 1;}return sockfd;
}
void do_wait(int signo)
{wait(NULL);
}
int main(int argc, char const *argv[])
{int sockfd = init_tcp("192.168.1.138", 50000);if (sockfd < 0){return 1;}signal(SIGCHLD, do_wait);char buf[1024] = {0};struct sockaddr_in cliaddr;int clilen = sizeof(cliaddr);while (1){int connfd = accept(sockfd, (struct sockaddr *)&cliaddr, &clilen);if (connfd < 0){perror("connect fail");return 1;}pid_t pid = fork();if (pid > 0){}else if (0 == pid){while (1){memset(buf, 0, sizeof(buf));ssize_t size = recv(connfd, buf, sizeof(buf), 0);if (size < 0){perror("recv fail");break;}else if (0 == size){printf("client connet offline");break;}printf("[%s] [%d]  : %s\n", inet_ntoa(cliaddr.sin_addr), ntohs(cliaddr.sin_port), buf);strcat(buf, "------ok");size = send(connfd, buf, sizeof(buf), 0);if (size < 0){perror("fail send");break;}}close(connfd);exit(1);}else{perror("fork fail");return 1;}}close(sockfd);return 0;
}

(2)多线程(并发程度高、不太安全)

#include "head.h"
int init_tcp(const char *ip, unsigned short port)
{int sockfd = socket(AF_INET, SOCK_STREAM, 0);if (sockfd < 0){perror("socket fail");return 1;}struct sockaddr_in seraddr;seraddr.sin_family = AF_INET;seraddr.sin_port = htons(port);seraddr.sin_addr.s_addr = inet_addr(ip);int ret = bind(sockfd, (struct sockaddr *)&seraddr, sizeof(seraddr));if (ret < 0){perror("bind fail");return 1;}ret = listen(sockfd, 100);if (ret < 0){perror("lisen fail");return 1;}return sockfd;
}
typedef struct
{int connfd;struct sockaddr_in cliaddr;
} XIN;void do_thurance(void *arg)
{XIN xi = *(XIN *)arg;char buf[1024] = {0};while (1){memset(buf, 0, sizeof(buf));ssize_t size = recv(xi.connfd, buf, sizeof(buf), 0);if (size < 0){perror("recv fail");break;}else if (0 == size){printf("client connet offline");break;}printf("[%s] [%d]  : %s\n", inet_ntoa(xi.cliaddr.sin_addr), ntohs(xi.cliaddr.sin_port), buf);strcat(buf, "------ok");size = send(xi.connfd, buf, sizeof(buf), 0);if (size < 0){perror("fail send");break;}}close(xi.connfd);pthread_exit(NULL);
}int main(int argc, char const *argv[])
{int sockfd = init_tcp("192.168.1.138", 50000);if (sockfd < 0){return 1;}char buf[1024] = {0};pthread_t tid;struct sockaddr_in cliaddr;int clilen = sizeof(cliaddr);while (1){int connfd = accept(sockfd, (struct sockaddr *)&cliaddr, &clilen);if (connfd < 0){perror("connect fail");return 1;}printf("client getline\n");XIN xi;xi.connfd = connfd;xi.cliaddr = cliaddr;pthread_create(&tid, NULL, do_thurance, &xi);pthread_detach(tid);                    //设置分离属性,线程结束,操作系统自动会回收;}close(sockfd);return 0;
}

(3)线程池

        主要解决:程序运行过程中,线程被反复创建和销毁带来的耗时问题;
(4)IO多路复用

        理解:不创建进程和线程的情况下,对多个文件描述符监测复用一个进程;

二、IO多路复用

1.阻塞IO方式:

(1)多个IO之间是同步关系;

(2)多个IO之间相互影响;

2.IO多路复用

(1)步骤
1)创建文件描述符集合(数组、链表、树形结构.......);

        2)添加关注的文件描述符带集合中;

        3)通过函数接口,把集合传递给内核,并开始检测IO事件(输入输出、读写事件);

        4)当内核检测到事件时,通过相关函数返回,做具体的相关操作;

(2)select
1)创建文件描述符集合表:fd_set

        2)清楚集合表     

        3)把文件描述符加入到集合表中

        4)select:   

        功能:通知内核检测的集合表并开始检测

        参数:

                nfds:关注的最大描述符+1

                readfds:关注的读事件的我文件描述符的地址

                writefds:关注的写事件的我文件描述符的地址

                exceptfds:其他事件

                timeout:超时事件的地址;设置一个时间结点,如果都没有事件来,就直接返回;        

                NULL:不设置超时时间

        返回值:

                成功:返回到达事件的个数

                失败:-1

                超时时间到达没有事件时:0

位图在内核中,保持最小未被使用原则

#include "head.h"
int init_tcp(const char *ip, unsigned short port)
{int sockfd = socket(AF_INET, SOCK_STREAM, 0);if (sockfd < 0){perror("socket fail");return 1;}struct sockaddr_in seraddr;seraddr.sin_family = AF_INET;seraddr.sin_port = htons(port);seraddr.sin_addr.s_addr = inet_addr(ip);int ret = bind(sockfd, (struct sockaddr *)&seraddr, sizeof(seraddr));if (ret < 0){perror("bind fail");return 1;}ret = listen(sockfd, 100);if (ret < 0){perror("lisen fail");return 1;}return sockfd;
}
int main(int argc, char const *argv[])
{int sockfd = init_tcp("192.168.1.138", 50002);if (sockfd < 0){return 1;}struct sockaddr_in cliaddr;int clilen = sizeof(cliaddr);int maxs;fd_set rdfds;fd_set tmprdfds;FD_ZERO(&rdfds);FD_SET(sockfd, &rdfds);int i = 0;maxs = sockfd;char buf[1024]={0};while (1){tmprdfds = rdfds;int cnt = select(maxs + 1, &tmprdfds, NULL, NULL, NULL);if (cnt < 0){perror("fail select");return 1;}if (FD_ISSET(sockfd, &tmprdfds)){int connfd = accept(sockfd,(struct sockaddr*)&cliaddr, &clilen);if (connfd < 0){perror("fail accept");return 1;}FD_SET(connfd, &rdfds);maxs = maxs > connfd ? maxs : connfd;}// for(i=sockfd;i<maxs+1;++i)// {//     printf("%d\n",i);// }// sleep(3);for (i = sockfd + 1; i < maxs + 1; ++i){if (FD_ISSET(i, &tmprdfds)){memset(buf, 0, sizeof(buf));ssize_t size = recv(i, buf, sizeof(buf), 0);if (size < 0){perror("recv fail");continue;}if(0==size){printf("client offlink\n");return 1;}printf("[%s] [%d]  : %s\n", inet_ntoa(cliaddr.sin_addr), ntohs(cliaddr.sin_port), buf);strcat(buf, "------ok");size = send(i, buf, sizeof(buf), 0);if (size < 0){perror("fail send");continue;}}  }}close(sockfd);return 0;
}

  缺陷:

        限制了最多只能检测1024个文件描述符(底层使用数组的机制储存);
在应用层每次需要遍历才可找到到达的事件的文件描述符,效率不高,还耗时;
集合表存在于应用层,内核存在应用层和内核层的数据表的反复拷贝,耗时;
select只能工作在水平触发模式(低速模式),不能工作在边沿触发模式(高速模式);

        边沿触发:数据从无变有,从低电平到高电平,触发一次,称读数据的上升沿触发;  数据一次收不完,但是下一次继续读

              水平触发:数据从无到有,先触发一次读,没读完,再触发读,一直到读完了,才不触发;优势在反复把数据读完;缺点:耗时,低俗模式

   (3)poll
1)解决的问题:检测的文件描述符个数不受1024限制;底层对于集合表的方式改变,变成了链表,时间复杂度O(n),其他问题未被改善,仍然需要反复拷贝、遍历、只可工作在水平触发模式;

(4)epoll
1)解决的问题:检测的文件描述符是树形结构;时间复杂度是O(log(n)【红黑树】,也不受1024限制;将检测的文件描述符集合创建在内核,解决了内核和用户层的数据拷贝;直接返回到达事件的文件描述符集合,不需要遍历寻找;epoll可以工作在水平触摸式,也可工作在边沿触发模式;

        2)步骤

                a)创建文件描述符集合表;       

                功能:创建文件描述符集合表到内核

                参数:

                        size:最多监测的文件描述符的个数

                返回值:

                        成功返回非负的文件描述符,代表了内核的集合;

                        失败返回-1

                b)添加关注的文件描述符到集合;

                int epoll_ctl(int epfd, int op, int fd, struct epoll_event *event);

                功能:对文件描述符进行操作;

                参数:

                        epfd:文件描述符集合表的文件描述符

                        op:        
EPOLL_CTL_ADD        新增事件
EPOLL_CTL_MOD        修改事件 
EPOLL_CTL_DEL        删除事件
fd:要操作的文件描述符 
events:事件相关结构体

                                         EPOLLIN        读事件
EPOLLOUT    写事件  
EPOLLET        边沿触发    
LT            水平触发

epoll函数里面的data里存放很多相关内容,需将fd直接放入此类型结构体里进行调用

                typedef union epoll_data {
void        *ptr;
int          fd;
uint32_t     u32;
uint64_t     u64;
} epoll_data_t;

                struct epoll_event {
uint32_t     events;      /* Epoll events */
epoll_data_t data;        /* User data variable */
};

                返回值:

                        成功返回0;

                        失败返回-1;

                c)epoll通知内核开始检测;

       int epoll_wait(int epfd, struct epoll_event *events,
int maxevents, int timeout);

                功能:监听事件表中的事件,并将产生的事件存放到结构体数组中
参数:
epfd:事件表文件描述符
events:存放结果事件结构体数组空间首地址 
maxevents:最多存放事件个数
timeout:超时时间
-1:阻塞等待直到有事件发生 
返回值:
成功返回产生事件个数
失败返回-1 

                  d)epoll返回检测到的事件结果

步骤:        1.创建文件描述符集合表;
2.添加关注的文件描述符到集合
3.epoll通知内核开始监测
4.epoll返回监测到的事件结果

3.在数据量比较小的时候,select的比epoll的性能差不多,甚至更好,更小的时候,优势体现不明显,对于IO:如果处理的任务有耗时任务,此时应该考虑增加进线程,把耗时的任务给进线程去做

并发函数对比

IO多路复用——并发服务器

并发服务器的性能对比

线程池+epoll 

对于IO:如果处理的任务有耗时任务,此时应该考虑增加进线程,把耗时的任务给进线程去做

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/news/919139.shtml
繁体地址,请注明出处:http://hk.pswp.cn/news/919139.shtml
英文地址,请注明出处:http://en.pswp.cn/news/919139.shtml

如若内容造成侵权/违法违规/事实不符,请联系英文站点网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

论文润色不能降低文章的重复率

最近大家问到多的&#xff0c;你们润色好了重复率会不会就降低了。这事儿啊&#xff0c;得从好几个方面去剖析&#xff0c;今天咱们就一块儿来探个究竟。咱们先得清楚&#xff0c;重复率检测工具一般会把内容标记成两类&#xff1a;一是那些和其他文献在文字表达上高度相似的部…

Python爬虫实战:构建alltheplaces平台地理信息数据采集系统

1. 引言 1.1 研究背景与意义 在大数据与智慧城市建设的推动下,地理位置信息(如餐馆、景点、公共设施等 POI 数据)已成为商业分析、城市规划、公共服务优化的核心基础数据。alltheplaces 作为全球领先的开放场所数据平台,整合了来自多个数据源的标准化信息,涵盖场所名称、…

HTML第三次作业

抽奖项目代码<!DOCTYPE html> <html lang"zh-CN"> <head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><title>简易抽奖转盘</title><sty…

PyTorch 面试题及详细答案120题(01-05)-- 基础概念与安装

《前后端面试题》专栏集合了前后端各个知识模块的面试题&#xff0c;包括html&#xff0c;javascript&#xff0c;css&#xff0c;vue&#xff0c;react&#xff0c;java&#xff0c;Openlayers&#xff0c;leaflet&#xff0c;cesium&#xff0c;mapboxGL&#xff0c;threejs&…

云手机选哪个比较好用?

云手机作为基于云计算技术运行的一款虚拟手机&#xff0c;能够帮助企业与个人用户进行账号多开和远程访问等多种功能&#xff0c;是手游玩家的首要选择&#xff0c;能够多开账号挂机不卡顿&#xff0c;但是哪一款云手机更加流畅好用呢&#xff1f;对于热衷于手游的玩家来说&…

[科研理论]无人机底层控制算法PID、LQR、MPC解析

文章目录1. PX4飞控PID简介1.1 位置控制器1.2 速度控制器1.3 加速度和yaw转到姿态1.4 姿态控制器1.5 角速率控制器2. 线性二次型优化&#xff08;LQR&#xff09;控制3. 模型预测控制MPC/NMPC3.1 MPC3.2 NMPC1. PX4飞控PID简介 相关链接&#xff1a;PX4官方中文文档、PID概念(…

AI系统性思维复盘概述

核心价值&#xff1a;从“被动思考”到“主动进化”。 基于数据驱动、机器学习和知识图谱的智能化组织学习系统&#xff0c;它将经验积累从传统的主观性、碎片化模式转变为客观性、系统化的科学模式&#xff0c;最终实现从被动应对向主动预防、从经验决策向数据决策、从个体智慧…

C++继承(2)

2.基类和派生类间的转换 •public继承的派⽣类对象可以赋值给基类的指针/基类的引⽤。这⾥有个形象的说法叫切⽚或者切 割。寓意把派⽣类中基类那部分切出来&#xff0c;基类指针或引⽤指向的是派⽣类中切出来的基类那部分。 • 基类对象不能赋值给派⽣类对象。 • 基类的指针或…

easya2a: 一键将 LangChain Agent 发布为 A2A 服务

easya2a: 一键将 LangChain Agent 发布为 A2A 服务 随着 A2A (Agent-to-Agent) 协议的发布&#xff0c;相关的实践项目也逐渐涌现。对于许多希望体验 A2A 功能&#xff0c;但又担心学习成本和开发时间的开发者来说&#xff0c;推荐使用 easya2a——一个可以快速、无缝地将现有 …

原学之设计模式- 设计模式来源

引言 各位旅行者们你们好&#xff0c;我是小森&#xff0c;首先我为啥是程序员。学了技术快六年了&#xff0c;但一直都是断断续续&#xff0c;本身自己的条件&#xff0c;从2021年11月份开始下载原神&#xff0c;总而言之不了解一些抽卡机制导致退了并且删除了具体账号打算重新…

有鹿机器人:AI技术如何重新定义「扫地」这件小事?

当扫地成为一门“技术活”扫地&#xff0c;可能是人类最古老的清洁行为之一。从扫帚到吸尘器&#xff0c;再到今天的无人驾驶清洁设备&#xff0c;我们一直在寻找更高效、更彻底的方式维护环境整洁。但有鹿机器人的出现&#xff0c;让“扫地”这件事有了新的定义——它不再只是…

62.不同路径

dp问题描述 62.不同路径 确定本题的状态表示 dp[i,j]表示的是从左上角走到这个位置的路径条数 确定本题的状态转移方程 根据已知条件&#xff1a;dp[0,0]1&#xff0c;dp[0,1]1&#xff0c;dp[1,0]1 本题的状态转移方程是&#xff1a; dp[i,j]dp[i,j-1]dp[i-1,j] 填表求…

python---包

文章目录包的基本概念创建包的基本结构__init__.py文件导入包和模块相对导入&#xff08;在包内部使用&#xff09;导入包和导入模块的区别包是Python中组织模块的一种方式&#xff0c;它允许你将相关的模块分组在一起&#xff0c;形成一个层次结构。包的主要目的是帮助避免命名…

超详细yolov8/11-obb旋转框全流程概述:配置环境、数据标注、训练、验证/预测、onnx部署(c++/python)详解

因为yolo的检测/分割/姿态/旋转/分类模型的环境配置、训练、推理预测等命令非常类似&#xff0c;这里不再详细叙述环境配置&#xff0c;主要参考【超详细yolo8/11-detect目标检测全流程概述&#xff1a;配置环境、数据标注、训练、验证/预测、onnx部署(c/python)详解】&#xf…

创世理论达成 全关联的动态振动网:量子世界的“底层逻辑”

全关联的动态振动网&#xff1a;量子世界的“底层逻辑”&#xff08;不带公式&#xff0c;超级详细&#xff09;要真正理解量子世界的本质&#xff0c;我们需要跳出“粒子”和“波”的传统框架&#xff0c;从量子场论的核心逻辑出发&#xff0c;用最生活化的类比和日常经验&…

银行间交易IMIX协议加密相关

加密流程 字段筛选与序列化 提取业务报文中标记为敏感的字段&#xff0c;生成待加密的数据块 <!-- 示例&#xff1a;原始交易指令 --> <Order><Symbol>ABC123</Symbol> <!-- 非敏感 --><Price>100.50</Price> …

ABM和强化学习-2015年全国大学生数学建模竞赛B题

多智能体系统&#xff08;Agent-Based Model, ABM&#xff09;和强化学习&#xff08;Reinforcement Learning, RL&#xff09;是两个不同但可结合的概念&#xff0c;尤其在复杂系统建模和人工智能领域有重要应用。下面分别解释它们&#xff0c;并说明二者的关联&#xff1a; …

ALBEF/BLIP/BLIP2/Instruct BLIP/X Instruct BLIP

ALBEF 研究动机 ALBEF之前的一些方式&#xff0c; 视觉分支基本都是基于 dector的方式&#xff08;检出目标框&#xff09;&#xff0c;如下图所示&#xff0c;由于大量的数据没有标框&#xff0c;因此视觉预训练的受限于dector的检测方式。创新点 能不能不再采用dector的方式训…

数据结构——排序算法(简单篇:冒泡排序、选择排序、插入排序)

1️⃣ 冒泡排序&#xff08;Bubble Sort&#xff09; 基本思想 重复地比较相邻的两个元素&#xff0c;如果顺序错误就交换它们。一趟冒泡结束后&#xff0c;最大&#xff08;或最小&#xff09;的元素会“浮”到末尾。下一趟时可以少比较一次&#xff0c;因为最后的元素已经排好…

配置 Docker 镜像加速,解决 docker pull 拉取镜像失败、docker search 查询镜像失败等问题

一、概述 记录时间 [2025-08-16] 在 Docker 学习中&#xff0c;可能会遇到诸如 docker 远程仓库无法访问、docker pull 拉取镜像失败、docker search 查询镜像失败等问题。 这是由于国内网络对 docker 远程仓库的访问受到限制。 那么在国内如何获取 docker 镜像呢&#xff1f…