当前位置: 首页 > article >正文

linux——TCP多线程并发服务器

多线程服务器 可以同时处理多个客户端旧版一次只能接一个客户客户不走别人连不进来。新版来一个客户创建一个线程专门服务同时支持 N 个客户端主函数加了一个while(1)循环pthread_t tid; int newfd -1; struct sockaddr_in cin; socklen_t addrlen sizeof(cin); while(1) { newfd accept(fd,(struct sockaddr *)cin,addrlen); if(newfd 0) { perror(accept); exit(1); } char ipv4_addr[16]; if(!inet_ntop(AF_INET,(void *)cin.sin_addr,ipv4_addr,sizeof(cin))) { perror(inet_ntop); exit(1); } printf(Client:(%s,%d) is connect\n,ipv4_addr,ntohs(cin.sin_port)); pthread_create(tid,NULL,client_data_handle,(void *)newfd); }就是无线循环accept函数accept 接到一个客户 → 立刻创建一个线程去服务它 → 主线程马上回去继续等下一个客户创建线程pthread_create(tid, NULL, client_data_handle, (void *)newfd);让这个线程去执行client_data_handle函数专门服务当前这个客户端主线程马上回去继续 accept 等下一个客户线程函数用来接收数据代码void* client_data_handle(void* arg) { int newfd *(int *)arg; char buf[BUFSIZE]; int ret -1; printf(handle thread :newfd %d\n,newfd); while(1) { do { bzero(buf,BUFSIZE); ret read(newfd,buf,BUFSIZE-1); }while(ret 1); if(ret 0) { exit(1); } if(!ret) { break; } printf(receive data:%s\n,buf); if(!strncasecmp(buf,QUIT_STR,strlen(QUIT_STR))) { printf(Client is exiting!\n); break; } } close(newfd); return NULL; }每个客户端都由一个独立的线程服务同时也把客户端代码优化一下以前的客户端代码IP地址和端口号是写死的我们改成命令行传参int main(int argc, char **argv) { port atoi(argv[2]); sin.sin_addr.s_addr inet_addr(argv[1]); }这样客户端更灵活相连接不同的服务器不用总是修改代码想连谁在终端输入对应的IP地址和端口号就行服务器端#includestdio.h #include sys/types.h /* See NOTES */ #include sys/socket.h #include unistd.h #includestdlib.h #include strings.h #include arpa/inet.h #includestring.h #include pthread.h #define QUIT_STR QUIT #define BUFSIZE 1024 #define BACKLOG 5 #define SERV_IP 5001 #define SERV_IP_ADDR 192.168.88.129 void* client_data_handle(void* arg); int main() { int fd -1; struct sockaddr_in sin; //1.socket fd socket(AF_INET,SOCK_STREAM,0); if(fd0) { perror(socket); exit(1); } bzero(sin,sizeof(sin)); sin.sin_family AF_INET; sin.sin_port htons(SERV_IP); //sin.sin_addr.s_addr inet_addr(SERV_IP_ADDR); sin.sin_addr.s_addr INADDR_ANY; /*if(inet_pion(AF_INET,SERV_IP_ADDR,(void *)sin.sin_addr.s_addr) ! 1) { perror(inet_pton); exit(1); } */ //2.bind if(bind(fd,(struct sockaddr *)sin,sizeof(sin)) 0) { perror(bind); exit(0); } //3.listen if(listen(fd,BACKLOG) 0) { perror(listen); exit(1); } //4.accept /*int newfd -1; newfd accept(fd,NULL,NULL); if(newfd 0) { perror(accept); exit(1); } */ pthread_t tid; int newfd -1; struct sockaddr_in cin; socklen_t addrlen sizeof(cin); while(1) { newfd accept(fd,(struct sockaddr *)cin,addrlen); if(newfd 0) { perror(accept); exit(1); } char ipv4_addr[16]; if(!inet_ntop(AF_INET,(void *)cin.sin_addr,ipv4_addr,sizeof(cin))) { perror(inet_ntop); exit(1); } printf(Client:(%s,%d) is connect\n,ipv4_addr,ntohs(cin.sin_port)); pthread_create(tid,NULL,client_data_handle,(void *)newfd); } close(fd); return 0; } void* client_data_handle(void* arg) { int newfd *(int *)arg; char buf[BUFSIZE]; int ret -1; printf(handle thread :newfd %d\n,newfd); while(1) { do { bzero(buf,BUFSIZE); ret read(newfd,buf,BUFSIZE-1); }while(ret 1); if(ret 0) { exit(1); } if(!ret) { break; } printf(receive data:%s\n,buf); if(!strncasecmp(buf,QUIT_STR,strlen(QUIT_STR))) { printf(Client is exiting!\n); break; } } close(newfd); return NULL; }客户端#includestdio.h #includestdlib.h #includestring.h #includestrings.h #includeunistd.h #include sys/types.h /* See NOTES */ #include sys/socket.h #include arpa/inet.h // ./client 192.168.88.129 5001 #define SERV_PORT 5001 #define SERV_IP_ADDR 192.168.88.129 #define BUFSIZE 1024 #define QUIT_STR QUIT int main(int argc,char **argv) { int fd -1; if(argc ! 3) { exit(1); } int port -1; port atoi(argv[2]); struct sockaddr_in sin; fd socket(AF_INET,SOCK_STREAM,0); if(fd 0) { perror(socket); exit(1); } bzero(sin,sizeof(sin)); sin.sin_family AF_INET; sin.sin_port htons(port); sin.sin_addr.s_addr inet_addr(argv[1]); if(connect(fd,(struct sockaddr*)sin,sizeof(sin)) 0) { perror(connect); exit(1); } char buf[BUFSIZE]; while(1) { bzero(buf,BUFSIZE); if(fgets(buf,BUFSIZE-1,stdin) NULL) { continue; } write(fd,buf,strlen(buf)); if(!strncasecmp(buf,QUIT_STR,strlen(QUIT_STR))) { break; } } return 0; }成功实现了

相关文章:

linux——TCP多线程并发服务器

多线程服务器 可以同时处理多个客户端旧版:一次只能接一个客户,客户不走,别人连不进来。新版:来一个客户,创建一个线程专门服务,同时支持 N 个客户端!主函数加了一个while(1)循环pthread_t tid…...

HL1606 LED灯带PWM驱动库:9/12/15位可配置灰度实现

1. HL1606 LED Strip PWM 库深度技术解析HL1606 是一款经典的串行级联LED驱动芯片,广泛应用于早期RGB LED灯带(如Adafruit早期的“NeoPixel前身”方案)。与WS2812B等单线协议芯片不同,HL1606采用标准SPI接口配合独立锁存信号&…...

从编译到实战:用MRtrix3处理你的第一份DWI数据(附macOS Ventura适配指南)

从编译到实战:用MRtrix3处理你的第一份DWI数据(附macOS Ventura适配指南) 第一次打开MRtrix3的命令行界面时,那种面对未知领域的兴奋与忐忑,相信每位神经影像研究者都深有体会。这个开源的弥散磁共振成像处理工具&…...

让开发流程更高效:为 Visual Studio 订阅用户解锁 Syncfusion苟

一、什么是requests? requests 是一个用于发送HTTP请求的 Python 库。 它可以帮助你: 轻松发送GET、POST、PUT、DELETE等请求 处理Cookie、会话等复杂性 自动解压缩内容 处理国际化域名和URL 二、应用场景 requests 广泛应用于以下实际场景: …...

记录复现多模态大模型论文OPERA的一周工作()杖

pagehelper整合 引入依赖com.github.pagehelperpagehelper-spring-boot-starter2.1.0compile编写代码 GetMapping("/list/{pageNo}") public PageInfo findAll(PathVariable int pageNo) {// 设置当前页码和每页显示的条数PageHelper.startPage(pageNo, 10);// 查询数…...

ADS126X高精度Δ-Σ ADC驱动开发与工业应用实战

1. ADS126X高精度Δ-Σ ADC驱动库深度解析:面向工业级嵌入式系统的底层实现与工程实践ADS126X系列是德州仪器(Texas Instruments)推出的24位、超低噪声、高集成度Δ-Σ模数转换器,涵盖ADS1262与ADS1263两款核心型号。该系列专为高…...

别再当‘炼丹’黑盒侠了!用Grad-CAM给你的PyTorch/TensorFlow模型做个‘X光’检查

深度解密Grad-CAM:像外科手术般精准剖析CNN决策逻辑 当你的图像分类模型在测试集上表现优异,却在真实场景中频频出错时,作为开发者的你是否感到困惑?我们常常陷入一个怪圈:模型准确率很高,却不知道它究竟&q…...

Triton + RISC-V忱

. GIF文件结构 相比于 WAV 文件的简单粗暴,GIF 的结构要精密得多,因为它天生是为了网络传输而设计的(包含了压缩机制)。 当我们用二进制视角观察 GIF 时,它是由一个个 数据块(Block) 组成的&…...

嵌入式按钮事件处理库:多类型去抖与状态机驱动设计

1. 项目概述 r89m Buttons 是一个面向嵌入式系统的轻量级、可移植按钮事件处理库,专为统一管理多种物理形态与电气特性的按钮输入而设计。其核心目标并非仅实现“按下/释放”电平检测,而是构建一套 事件驱动的抽象层 ,将底层硬件差异&…...

CCC3.0数字钥匙系统架构解析:从蓝牙OOB配对到多设备互操作性

1. 从机械钥匙到数字钥匙的技术演进 记得十年前我第一次买车时,销售递给我的是一把沉甸甸的机械钥匙,上面还挂着一个印着品牌logo的钥匙扣。那时候根本想不到,短短几年后我们就能用手机解锁汽车。这种变化背后,是CCC(C…...

MATLAB+CPLEX仿真平台下的微网虚拟电厂日前优化调度模型:融合电动汽车出行及充放电规律...

MATLAB代码:含多种需求响应及电动汽车的微网/虚拟电厂日前优化调度 关键词:需求响应 空调负荷 电动汽车 微网优化调度 虚拟电厂调度 仿真平台:MATLABCPLEX 主要内容:代码主要做的是一个微网/虚拟电厂的日前优化调度模型&#…...

STM32duino VL53L0X驱动深度解析:ToF传感器嵌入式实践指南

1. STM32duino VL53L0X 库深度解析:面向嵌入式工程师的ToF传感器驱动实践指南VL53L0X 是意法半导体(STMicroelectronics)推出的第二代飞行时间(Time-of-Flight, ToF)激光测距传感器,采用940nm不可见红外VCS…...

新手入门RTOS,别再纠结了!从RT-Thread和FreeRTOS的实战项目选择说起

新手入门RTOS:从实战项目看RT-Thread与FreeRTOS的选择策略 第一次接触实时操作系统(RTOS)时,面对众多选择往往会感到迷茫。作为嵌入式开发领域的核心技术之一,RTOS的选择直接影响着项目的开发效率和最终性能表现。在众…...

Vue中手动取消watch监听的最佳实践与实现原理

1. 为什么需要手动取消watch监听 在Vue开发中,watch监听器是我们常用的响应式工具之一。它能够监听数据变化并执行相应的回调函数。但很多开发者可能没有意识到,不当管理watch监听器可能会导致内存泄漏和性能问题。 想象一下这样的场景:你在一…...

BigEarthNet-MM数据集太大跑不动?教你用TFRecord分片和增量处理加速实验

BigEarthNet-MM数据集优化处理实战:分片技术与增量加载全解析 当你的GPU风扇开始发出直升机般的轰鸣,而TensorFlow进度条像树懒散步一样缓慢时——这可能是BigEarthNet-MM数据集在提醒你:传统的全量加载方式已经不适合这个时代了。本文将带你…...

数据摄取构建模块简介(预览版)(一)弛

一、语言特性:Java 26 与模式匹配进化 1.1 Java 26 语言级别支持 IDEA 2026.1 EAP 最引人注目的变化之一,就是新增 Java 26 语言级别支持。这意味着开发者可以提前体验和测试即将在 JDK 26 中正式发布的语言特性。 其中最重要的变化是对 JEP 530 的全面支…...

教育部:加快普及中小学生人工智能教育政策汇总

教育部:加快普及中小学生人工智能教育政策汇总 基本信息 发布时间:2026-04-10(最新政策)政策文件:《"人工智能教育"行动计划》发文机构:教育部、国家发展改革委、工业和信息化部、科技部、国家…...

从“单细胞”到“多细胞”:MetaGPT、AutoGen、AgentVerse如何重塑AI应用开发范式?

从“单细胞”到“多细胞”:MetaGPT、AutoGen、AgentVerse如何重塑AI应用开发范式? 想象一下,当你对AI说"开发一个电商网站"时,不再只是得到零散的代码片段,而是一个完整的数字团队自动分工协作:产…...

Adafruit Protomatter:HUB75 LED矩阵的裸机GPIO驱动原理与实践

1. Adafruit Protomatter 库深度技术解析:面向 HUB75 RGB LED 矩阵的裸机 GPIO 驱动框架 1.1 核心定位与工程目标 Adafruit Protomatter 是一个专为驱动 HUB75 接口 RGB LED 矩阵而设计的轻量级、高可移植性底层库。其核心设计哲学并非追求极致性能,而是…...

保姆级教程:在Jetson Orin上从零搭建PyTorch+TensorFlow环境(含torchvision源码编译避坑)

保姆级教程:在Jetson Orin上从零搭建PyTorchTensorFlow环境(含torchvision源码编译避坑) NVIDIA Jetson Orin作为当前边缘计算领域的旗舰平台,其ARM架构下的深度学习环境配置一直是开发者的痛点。本文将手把手带你完成从系统准备到…...

字符串拼接用“+”还是 StringBuilder?别再凭感觉写了品

前言 Kubernetes 本身并不复杂,是我们把它搞复杂的。无论是刻意为之还是那种虽然出于好意却将优雅的原语堆砌成 鲁布戈德堡机械 的狂热。平台最初提供的 ReplicaSets、Services、ConfigMaps,这些基础组件简单直接,甚至显得有些枯燥。但后来我…...

浅谈MIKE前处理中投影坐标处理问题

MIKE 中投影坐标一直是个问题,尤其对 2d 里的科氏力影响很大, 由于我们现获取基础资料都是 CAD 格式,在 GIS 里转 shp 后我们会发现很多是地方坐标,对于这种情况,小编也是无能无力,只有想办法 让 CAD 提供方…...

智慧树自动刷课终极解决方案:5分钟告别手动刷课的完整指南

智慧树自动刷课终极解决方案:5分钟告别手动刷课的完整指南 【免费下载链接】zhihuishu 智慧树刷课插件,自动播放下一集、1.5倍速度、无声 项目地址: https://gitcode.com/gh_mirrors/zh/zhihuishu 还在为智慧树平台繁琐的网课学习而烦恼吗&#x…...

RAG分块策略实战:5种方法代码对比+真实业务场景选择指南(附性能测试数据)

RAG分块策略工程实践:5种方法性能对比与场景化选型指南 在构建检索增强生成(RAG)系统时,文档分块策略的选择直接影响着系统的最终效果。本文将深入分析五种主流分块策略的工程实现差异,结合电商客服、医疗问答等典型业…...

麒麟V10系统下微信PC版安装与系统升级全攻略

1. 麒麟V10系统与微信PC版适配现状 最近两年国产操作系统发展迅猛,银河麒麟V10作为其中的佼佼者,已经能够流畅运行微信PC版。但很多用户在安装过程中还是会遇到各种"拦路虎"——找不到安装包、依赖缺失、版本冲突等问题层出不穷。 我实测发现&…...

PX4 EKF滤波效果不好?别只盯着Q和R,这些隐藏参数和传感器预处理同样关键

PX4 EKF滤波效果优化:超越Q/R矩阵的隐藏参数与传感器预处理全解析 当你的无人机在悬停时出现位置漂移,或是穿越机在高速机动时姿态突然发散,大多数开发者第一反应就是调整Q和R矩阵——这就像医生遇到发烧就开退烧药,却忽略了病灶本…...

人工智能工程师应掌握的核心技能与工具

随着人工智能(AI)领域的持续拓展,对专业 AI 工程师的需求呈指数级增长。无论你是刚入行,还是希望实现职业进阶,扎实掌握特定技能与工具都至关重要。本文将详解每位 AI 工程师想要在这一充满活力且竞争激烈的领域立足所…...

OFDRW 2.1.0转换PDF时字体丢失?3种实用解决方案帮你搞定

OFDRW 2.1.0转换PDF字体丢失问题深度解析与实战解决方案 在企业级文档处理系统中,OFD(Open Fixed-layout Document)与PDF之间的格式转换是常见需求。作为国内电子发票、公文交换的标准格式,OFD的准确转换直接关系到业务数据的完整…...

深入剖析Ultralytics中RT-DETR的RepC3模块维度匹配问题

1. RT-DETR与RepC3模块的核心作用 RT-DETR作为Ultralytics推出的实时目标检测模型,其核心优势在于将DETR系列模型的Transformer架构与实时推理需求相结合。我在实际部署中发现,RepC3模块作为模型颈部的关键组件,承担着多尺度特征融合与通道维…...

M5StamPLC工业PLC库:ESP32嵌入式实时控制与I²C外设驱动

1. M5StamPLC 库概述M5StamPLC 是专为 M5Stack 推出的 K141 型号工业级可编程逻辑控制器(PLC)开发板设计的底层驱动库。该板卡并非传统意义上的 Arduino 兼容开发板,而是面向工业自动化场景的嵌入式控制终端,具备数字量输入/输出、…...