linux多线程并发服务器

我就是我 2022-11-29 13:29 238阅读 0赞

linux多线程并发服务器

此为多线程并发服务器,多进程版。

  1. #include <stdio.h>
  2. #include <unistd.h>
  3. #include <stdlib.h>
  4. #include <sys/types.h>
  5. #include <string.h>
  6. #include <sys/socket.h>
  7. #include <arpa/inet.h>
  8. #include <ctype.h>
  9. #include <pthread.h>
  10. //自定义数据结构
  11. typedef struct SockInfo
  12. {
  13. int fd;//文件描述符cfd
  14. struct sockaddr_in addr;//存放ip地址的结构体
  15. pthread_t id;//线程的id
  16. }SockInfo;
  17. //子线程处理函数
  18. void* worker(void* arg){
  19. char ip[64];
  20. char buf[1024];
  21. SockInfo* info = (SockInfo*)arg;
  22. //通信
  23. while(1)
  24. {
  25. printf("client IP:%s,port:%d\n",
  26. inet_ntop(AF_INET,&info->addr.sin_addr.s_addr,ip,sizeof(ip)),
  27. ntohs(info->addr.sin_port));
  28. int len = read(info->fd,buf,sizeof(buf));
  29. if(len == -1){
  30. perror("read error");
  31. pthread_exit(NULL);
  32. }
  33. else if(len ==0){
  34. printf("客户端断开了连接\n");
  35. close(info->fd);
  36. break;
  37. }
  38. else{
  39. printf("recv buf: %s\n",buf);
  40. write(info->fd,buf,len);
  41. }
  42. }
  43. return NULL;
  44. }
  45. int main(int argc, char const *argv[])
  46. {
  47. if(argc<2){
  48. printf("eg: ./a.out port\n");
  49. exit(1);
  50. }
  51. struct sockaddr_in serv_addr;
  52. socklen_t serv_len = sizeof(serv_addr);
  53. int port = atoi(argv[1]);
  54. //创建套接字
  55. int lfd = socket(AF_INET,SOCK_STREAM,0);
  56. //初始化服务器 sockaddr_in
  57. memset(&serv_addr,0,serv_len);
  58. serv_addr.sin_family = AF_INET; //地址族
  59. serv_addr.sin_addr.s_addr = htonl(INADDR_ANY); //监听本机所有的IP
  60. serv_addr.sin_port = htons(port); //设置端口
  61. //绑定IP和端口
  62. bind(lfd,(struct sockaddr*)&serv_addr,serv_len);
  63. //设置同时监听的最大个数
  64. listen(lfd,36);
  65. printf("Start accept ......\n");
  66. int i = 0;
  67. SockInfo info[256];//线程个数,限制最多为256
  68. //规定 数组里边的fd等于-1,表示当前结构体空闲
  69. for(i=0;i<sizeof(info)/sizeof(info[0]);++i){
  70. info[i].fd = -1;
  71. }
  72. struct sockaddr_in client_addr;
  73. socklen_t cli_len = sizeof(struct sockaddr_in);
  74. while(1){
  75. //选一个没有被使用的,最小的数组元素
  76. for(i=0;i<256;++i){
  77. if(info[i].fd == -1){
  78. break;
  79. }
  80. }
  81. if(i == 256){
  82. break;
  83. }
  84. //主线程 - 等待接受连接请求
  85. info[i].fd = accept(lfd,(struct sockaddr*)&info[i].addr,&cli_len);
  86. //创建子线程 - 通信
  87. pthread_create(&info[i].id,NULL,worker,&info[i]);
  88. //设置线程分离
  89. pthread_detach(info[i].id);
  90. }
  91. close(lfd);
  92. //只退出主线程
  93. pthread_exit(NULL);
  94. return 0;
  95. }

终端1

  1. $ gcc pthread_server.c -lpthread -o server
  2. $ ./server 9876
  3. Start accept ......
  4. client IP:127.0.0.1,port:39808
  5. recv buf: 321
  6. client IP:127.0.0.1,port:39808
  7. client IP:127.0.0.1,port:39810
  8. recv buf: 456
  9. client IP:127.0.0.1,port:39810
  10. client IP:127.0.0.1,port:39820
  11. recv buf: hahaha
  12. client IP:127.0.0.1,port:39820
  13. 客户端断开了连接
  14. 客户端断开了连接
  15. 客户端断开了连接

终端2

  1. $ nc 127.1 9876
  2. 321 -------------客户端输入的
  3. 321 -------------服务端返回的

终端3

  1. $ nc 127.1 9876
  2. 456 -------------客户端输入的
  3. 456 -------------服务端返回的

终端4

  1. $ nc 127.1 9876
  2. hahaha
  3. hahaha

发表评论

表情:
评论列表 (有 0 条评论,238人围观)

还没有评论,来说两句吧...

相关阅读

    相关 线并发

    1.并发就是对共享资源同时进行访问,而产生的现象。 2.线程的实现方式:继承Thread或实现runnable 都要重写run方法。 3.启动线程是通过Thread的sta