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

网络编程day3

进程与线程实现TCP并发服务器

多进程服务器端

#include<myhead.h>#define  SER_PORT  8888        //服务器端口号
#define SER_IP  "192.168.109.125"   //服务器IP地址 //定义一个信号处理函数,用于处理SIGCHLD信号
void handler(int signo)
{//判断是哪个信号到位if(signo == SIGCHLD){//说明有子进程牺牲了//使用非阻塞的形式循环回收子进程资源while(waitpid(-1, NULL, WNOHANG) > 0);}
}int main(int argc, const char *argv[])
{//将子进程向父进程发送的SIGCHLD信号,绑定到信号处理函数中if(signal(SIGCHLD, handler) == SIG_ERR){perror("signal error");return -1;}//1、创建一个用于连接的套接字文件描述符int sfd  = socket(AF_INET, SOCK_STREAM, 0);if(-1 == sfd){perror("socket  error");return -1;}printf("socket 成功 sfd = %d\n", sfd);          //3//设置套接字属性:允许端口号快速重用int reuse = 1;if(setsockopt(sfd, SOL_SOCKET, SO_REUSEADDR, &reuse, sizeof(reuse))==-1){perror("setsockopt error");return -1;}printf("端口号快速重用成功\n");//2、给套接字绑定ip地址和端口号//2.1 封装地址信息结构体变量struct sockaddr_in sin;        //地址信息结构体变量sin.sin_family = AF_INET;      //通信域地址族sin.sin_port = htons(SER_PORT);   //端口号网络字节序sin.sin_addr.s_addr = inet_addr(SER_IP);  //ip地址网络字节序//2.2  绑定操作if(bind(sfd, (struct sockaddr*)&sin, sizeof(sin)) ==-1){perror("bind  error");return -1;}printf("bind success\n");//3、将套接字启动监听if(listen(sfd, 128) == -1){perror("listen  error");return -1;}printf("listen success\n");//4、阻塞等等客户端的连接,如果有新客户端连接,//则创建一个用于通信的套接字struct sockaddr_in cin;            //用于接受客户端套接字信息socklen_t addrlen = sizeof(cin);   //用于接受客户端套接字的长度while(1)              //循环处理多个客户端{//接收新的客户端的连接请求int new_fd = accept(sfd, (struct sockaddr*)&cin, &addrlen);if(-1 == new_fd){perror("accept error");return -1;}printf("[%s:%d]发来连接,new_fd = %d\n", inet_ntoa(cin.sin_addr),\ntohs(cin.sin_port) ,new_fd);//给当前新的套接字创建一个子进程用于跟客户端通信pid_t pid = fork();if(pid > 0){//关闭new_fdclose(new_fd);}else if(pid == 0){//关闭sfd文件描述符close(sfd);//5、使用新套接字跟客户端进行通信while(1){//从套接字中读取消息char rbuf[128] = "";          //存放接受消息的容器//int res = read(new_fd, rbuf, sizeof(rbuf)-1);int res = recv(new_fd, rbuf, sizeof(rbuf)-1, 0);if(res == 0){printf("客户端已下线\n");close(new_fd);exit(EXIT_SUCCESS);           //退出子进程}if(res == -1){perror("read error");close(new_fd);close(sfd);return -1;}//如果程序执行至此,表示数据成功读取下来,病放入rbuf中printf("[%s:%d] : %s\n", inet_ntoa(cin.sin_addr), ntohs(cin.sin_port), rbuf);//给当前消息加个笑脸还回去strcat(rbuf, "*_*");//向套接字中写入消息//write(new_fd, rbuf, strlen(rbuf));send(new_fd, rbuf, strlen(rbuf), 0);printf("发送成功\n");}}//回收子进程资源}//6、关闭监听套接字close(sfd);return 0;
}

多线程客户端

#include<myhead.h>#define  SER_PORT  8888        //服务器端口号
#define SER_IP  "192.168.109.125"   //服务器IP地址 //定义一个结构体类型,用于主线程向分支线程传输多个数据
struct MsgInfo
{int new_fd;              //用于跟客户端通信的套接字文件描述符struct sockaddr_in cin;     //客户端套接字地址信息结构体
};//定义处理客户端消息的线程体函数
void *deal_cli_msg(void *arg)
{//将主线程传入的数据解析出来int new_fd = ((struct MsgInfo*)arg)->new_fd;struct sockaddr_in cin =  ((struct MsgInfo*)arg)->cin;//5、使用新套接字跟客户端进行通信while(1){//从套接字中读取消息char rbuf[128] = "";          //存放接受消息的容器//int res = read(new_fd, rbuf, sizeof(rbuf)-1);int res = recv(new_fd, rbuf, sizeof(rbuf)-1, 0);if(res == 0){printf("客户端已下线\n");close(new_fd);break;}if(res == -1){perror("read error");close(new_fd);return NULL;}//如果程序执行至此,表示数据成功读取下来,病放入rbuf中printf("[%s:%d] : %s\n", inet_ntoa(cin.sin_addr), ntohs(cin.sin_port), rbuf);//给当前消息加个笑脸还回去strcat(rbuf, "*_*");//向套接字中写入消息//write(new_fd, rbuf, strlen(rbuf));send(new_fd, rbuf, strlen(rbuf), 0);printf("发送成功\n");}//退出线程pthread_exit(NULL);}/*****************************主线程********************/
int main(int argc, const char *argv[])
{//1、创建一个用于连接的套接字文件描述符int sfd  = socket(AF_INET, SOCK_STREAM, 0);if(-1 == sfd){perror("socket  error");return -1;}printf("socket 成功 sfd = %d\n", sfd);          //3//设置套接字属性:允许端口号快速重用int reuse = 1;if(setsockopt(sfd, SOL_SOCKET, SO_REUSEADDR, &reuse, sizeof(reuse))==-1){perror("setsockopt error");return -1;}printf("端口号快速重用成功\n");//2、给套接字绑定ip地址和端口号//2.1 封装地址信息结构体变量struct sockaddr_in sin;        //地址信息结构体变量sin.sin_family = AF_INET;      //通信域地址族sin.sin_port = htons(SER_PORT);   //端口号网络字节序sin.sin_addr.s_addr = inet_addr(SER_IP);  //ip地址网络字节序//2.2  绑定操作if(bind(sfd, (struct sockaddr*)&sin, sizeof(sin)) ==-1){perror("bind  error");return -1;}printf("bind success\n");//3、将套接字启动监听if(listen(sfd, 128) == -1){perror("listen  error");return -1;}printf("listen success\n");//4、阻塞等等客户端的连接,如果有新客户端连接,//则创建一个用于通信的套接字struct sockaddr_in cin;            //用于接受客户端套接字信息socklen_t addrlen = sizeof(cin);   //用于接受客户端套接字的长度while(1){//阻塞等等客户端的连接//对于accept函数而言,当执行到该函数时,即使没有客户端的连接//也会预选一个当前最小的文件描述符给新套接字//在阻塞过程中,即使有更小的,也不会选择,会留作下一次预选使用int new_fd = accept(sfd, (struct sockaddr*)&cin, &addrlen);if(-1 == new_fd){perror("accept error");return -1;}printf("[%s:%d]发来连接,new_fd = %d\n", inet_ntoa(cin.sin_addr),\ntohs(cin.sin_port) ,new_fd);//定义一个结构体变量,存储通信的套接字文件描述符和客户端地址信息结构体struct MsgInfo info = {new_fd, cin};//创建一个分之现场,用于跟客户端进行通信pthread_t tid = -1;if(pthread_create(&tid, NULL, deal_cli_msg, &info)!=0){printf("线程创建失败\n");return -1;}//将线程设置成分离态pthread_detach(tid);}//6、关闭监听套接字close(sfd);return 0;
}

http://www.xdnf.cn/news/18194.html

相关文章:

  • leetcode2248. 多个数组求交集
  • Android13车机系统自定义系统栏显示策略之状态栏下拉异常
  • java八股文-中间件-参考回答
  • Commons-io
  • 微算法科技(NASDAQ: MLGO)研究利用PBFT中的动态视图变换机制,实现区块链系统高效运转
  • 2025年5月架构设计师综合知识真题回顾,附参考答案、解析及所涉知识点(六)
  • 笔试——Day43
  • HJ4 字符串分隔
  • C++高频知识点(二十七)
  • CentOS安装SNMPWalk
  • 无畏契约手游上线!手机远控模拟器畅玩、抢先注册稀有ID!
  • Linux的基本操作
  • 遥感amp;机器学习入门实战教程 | Sklearn 案例③:PCA + SVM / 随机森林 对比与调参
  • LAMP架构编译安装部署
  • 垂直领域大模型构建:法律行业“类ChatGPT”系统的训练与落地
  • PythonDay31
  • Vue2+Vue3前端开发_Day1
  • Fragment重要知识点总结
  • Incredibuild 新增 Unity 支持:击破构建时间过长的痛点
  • 机器学习(决策树2)
  • MVVM开源项目
  • Netty处理粘包与拆包
  • vue使用vue-cropper实现图片裁剪之单图裁剪
  • 关于mybatis表关联查询和mybatis-Plus单表查询传入时间查询数据(走索引)
  • Linux Namespace 隔离的“暗面”——故障排查、认知误区与演进蓝图
  • CVPR 2025 | 具身智能 | HOLODECK:一句话召唤3D世界,智能体的“元宇宙练功房”来了
  • 【HTML】3D动态凯旋门
  • 通过C#上位机串口写入和读取浮点数到stm32的片内flash实战4(通过串口下发AD9833设置值并在上位机显示波形曲线)
  • “你不干有的是AI干”,提示词中的“情感化提示”
  • 如何在 Ubuntu Linux 上安装 RPM 软件包