基于TCP客户端和服务器的I/O多路复用

一、基于TCP/IP协议的基本循环服务器

tcp_server.c

[cpp]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. #include <stdio.h>  
  2. #include <stdlib.h>  
  3. #include <string.h>  
  4.   
  5. #include <sys/types.h>  
  6. #include <sys/socket.h>  
  7. #include <sys/stat.h>  
  8. #include <netdb.h>  
  9.   
  10. #define PORT 3333  
  11. #define MAX_SIZE 1024  
  12.   
  13. int main()  
  14. {  
  15.     int sockfd;  
  16.     int new_fd;  
  17.   
  18.     struct sockaddr_in server_addr;  
  19.     struct sockaddr_in client_addr;  
  20.       
  21.     int n_read;  
  22.     int ser_size;  
  23.     int opt = 1;  
  24.   
  25.     char buffer[MAX_SIZE];  
  26.   
  27.     if((sockfd = socket(AF_INET,SOCK_STREAM,0)) < 0)  
  28.     {  
  29.         perror("socket error!\n");  
  30.     exit(1);  
  31.     }  
  32.   
  33.     printf("socket success.............!\n");  
  34.   
  35.     if(setsockopt(sockfd,SOL_SOCKET,SO_REUSEADDR,&opt,sizeof(opt)) < 0)  
  36.     {  
  37.         perror("set socket option error!\n");  
  38.     exit(1);  
  39.     }  
  40.   
  41.     bzero(&server_addr,0);  
  42.     server_addr.sin_family = AF_INET;  
  43.     server_addr.sin_port = htons(PORT);  
  44.     server_addr.sin_addr.s_addr = inet_addr("192.168.1.10");   
  45.   
  46.     if(bind(sockfd,(struct sockaddr *)&server_addr,sizeof server_addr) < 0)  
  47.     {  
  48.         perror("bind error!\n");  
  49.     exit(1);  
  50.     }  
  51.   
  52.     printf("bind success.............!\n");  
  53.     listen(sockfd,5);  
  54.        
  55.     printf("listen success.............!\n");  
  56.   
  57.     printf("accepting..................!\n");  
  58.       
  59.     while(1)  
  60.     {  
  61.         ser_size = sizeof client_addr;  
  62.   
  63.         if((new_fd = accept(sockfd,(struct sockaddr *)&client_addr,&ser_size)) < 0)  
  64.         {  
  65.             perror("accept error!\n");  
  66.         exit(1);  
  67.         }  
  68.   
  69.         printf("accept success.................!\n");  
  70.   
  71.     //read\recv  
  72.           
  73.         printf("reading............!\n");  
  74.         n_read = read(new_fd,buffer,sizeof(buffer));  
  75.   
  76.         if(n_read < 0)  
  77.         {  
  78.             perror("read client msg error!\n");  
  79.         exit(1);  
  80.         }  
  81.   
  82.         buffer[n_read] = '\0';  
  83.   
  84.         printf("recv msg = %s\n",buffer);  
  85.     }  
  86.   
  87.     return 0;  
  88. }  
tcp_client.c

[cpp]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. #include <stdio.h>  
  2. #include <string.h>  
  3. #include <stdlib.h>  
  4.   
  5. #include <sys/types.h>  
  6. #include <sys/socket.h>  
  7. #include <netdb.h>  
  8.   
  9. #define PORT 3333  
  10. #define MAX_SIZE 1024  
  11.   
  12. int main(int argc, char *argv[])  
  13. {  
  14.     if(argc != 2)  
  15.     {  
  16.         printf("Please input server ip!\n");  
  17.     exit(1);  
  18.     }  
  19.       
  20.     int sockfd;  
  21.     int n_write;  
  22.       
  23.     struct sockaddr_in server_addr;  
  24.   
  25.     char buffer[MAX_SIZE];  
  26.   
  27.     if((sockfd = socket(AF_INET,SOCK_STREAM,0)) < 0)  
  28.     {  
  29.         perror("client socket error!\n");  
  30.     exit(1);  
  31.     }  
  32.   
  33.     bzero(&server_addr,0);  
  34.   
  35.     server_addr.sin_family = AF_INET;  
  36.     server_addr.sin_port = htons(PORT);  
  37.     server_addr.sin_addr.s_addr = inet_addr(argv[1]);  
  38.   
  39.     if(connect(sockfd,(struct sockaddr *)&server_addr,sizeof server_addr) < 0)  
  40.     {  
  41.         perror("connect server error!\n");  
  42.     exit(1);  
  43.     }  
  44.       
  45.     while(1)  
  46.     {  
  47.         memset(buffer,0,sizeof(buffer));  
  48.     printf("Please input send msg:\n");  
  49.         gets(buffer);  
  50.   
  51.         n_write = write(sockfd,buffer,strlen(buffer));  
  52.   
  53.     if(n_write == -1)  
  54.     {  
  55.         perror("send to server msg error!\n");  
  56.         exit(1);  
  57.     }  
  58.     }  
  59.       
  60.     return 0;  
  61. }  
循环服务器能不断地接收客户端发送过来的信息,但是在面对多个客户端时,服务器必须等待第一个客户端发送信息才能接受其他客户端发来的信息。

由此,我们使用并发服务器,通过创建子进程,使得多个客户端能随时发信息给服务器

并发服务器:

[cpp]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. #include <stdio.h>  
  2. #include <stdlib.h>  
  3. #include <string.h>  
  4.   
  5. #include <sys/types.h>  
  6. #include <sys/socket.h>  
  7. #include <sys/stat.h>  
  8. #include <netdb.h>  
  9.   
  10. #define PORT 3333  
  11. #define MAX_SIZE 1024  
  12.   
  13. void * read_msg(void *arg)  
  14. {  
  15.     int n_read;  
  16.       
  17.     int new_fd = *((int *)arg);  
  18.       
  19.     char buffer[MAX_SIZE];  
  20.   
  21.     printf("new_fd = %d\n",new_fd);  
  22.   
  23.     while(1)  
  24.     {  
  25.         memset(buffer,0,sizeof(buffer));  
  26.   
  27.     n_read = read(new_fd,buffer,sizeof(buffer));  
  28.   
  29.     if(n_read < 0)  
  30.     {  
  31.         perror("recv client msg error!\n");  
  32.         exit(1);  
  33.     }  
  34.   
  35.     if(n_read == 0)  
  36.     {  
  37.         printf("client is close!!\n");  
  38.         close(new_fd);  
  39.         pthread_exit(NULL);  
  40.     }  
  41.   
  42.     buffer[n_read] = '\0';  
  43.   
  44.     printf("recv msg:%s\n",buffer);  
  45.     }  
  46. }  
  47.   
  48. int main()  
  49. {  
  50.     int sockfd;  
  51.     int new_fd;  
  52.   
  53.     struct sockaddr_in server_addr;  
  54.     struct sockaddr_in client_addr;  
  55.       
  56.     int n_read;  
  57.     int ser_size;  
  58.     int opt = 1;  
  59.   
  60.     char buffer[MAX_SIZE];  
  61.   
  62.     pthread_t id;  
  63.   
  64.     if((sockfd = socket(AF_INET,SOCK_STREAM,0)) < 0)  
  65.     {  
  66.         perror("socket error!\n");  
  67.     exit(1);  
  68.     }  
  69.   
  70.     printf("socket success.............!\n");  
  71.   
  72.     if(setsockopt(sockfd,SOL_SOCKET,SO_REUSEADDR,&opt,sizeof(opt)) < 0)  
  73.     {  
  74.         perror("set socket option error!\n");  
  75.     exit(1);  
  76.     }  
  77.   
  78.     bzero(&server_addr,0);  
  79.     server_addr.sin_family = AF_INET;  
  80.     server_addr.sin_port = htons(PORT);  
  81.     server_addr.sin_addr.s_addr = inet_addr("192.168.1.10");   
  82.   
  83.     if(bind(sockfd,(struct sockaddr *)&server_addr,sizeof server_addr) < 0)  
  84.     {  
  85.         perror("bind error!\n");  
  86.     exit(1);  
  87.     }  
  88.   
  89.     printf("bind success.............!\n");  
  90.     listen(sockfd,5);  
  91.        
  92.     printf("listen success.............!\n");  
  93.   
  94.     printf("accepting..................!\n");  
  95.       
  96.     while(1)  
  97.     {  
  98.         ser_size = sizeof client_addr;  
  99.   
  100.         if((new_fd = accept(sockfd,(struct sockaddr *)&client_addr,&ser_size)) < 0)  
  101.         {  
  102.             perror("accept error!\n");  
  103.         exit(1);  
  104.         }  
  105.   
  106.         printf("accept success.................!\n");  
  107.           
  108.     pthread_create(&id,NULL,read_msg,&new_fd);  
  109.     }  
  110.   
  111.     return 0;  
  112. }  
并发客户端:

[cpp]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. #include <stdio.h>  
  2. #include <string.h>  
  3. #include <stdlib.h>  
  4.   
  5. #include <sys/types.h>  
  6. #include <sys/socket.h>  
  7. #include <netdb.h>  
  8.   
  9. #define PORT 3333  
  10. #define MAX_SIZE 1024  
  11.   
  12. int main(int argc, char *argv[])  
  13. {  
  14.     if(argc != 2)  
  15.     {  
  16.         printf("Please input server ip!\n");  
  17.     exit(1);  
  18.     }  
  19.       
  20.     int sockfd;  
  21.     int n_write;  
  22.       
  23.     struct sockaddr_in server_addr;  
  24.   
  25.     char buffer[MAX_SIZE];  
  26.   
  27.     if((sockfd = socket(AF_INET,SOCK_STREAM,0)) < 0)  
  28.     {  
  29.         perror("client socket error!\n");  
  30.     exit(1);  
  31.     }  
  32.   
  33.     bzero(&server_addr,0);  
  34.   
  35.     server_addr.sin_family = AF_INET;  
  36.     server_addr.sin_port = htons(PORT);  
  37.     server_addr.sin_addr.s_addr = inet_addr(argv[1]);  
  38.   
  39.     if(connect(sockfd,(struct sockaddr *)&server_addr,sizeof server_addr) < 0)  
  40.     {  
  41.         perror("connect server error!\n");  
  42.     exit(1);  
  43.     }  
  44.       
  45.     while(1)  
  46.     {  
  47.         memset(buffer,0,sizeof(buffer));  
  48.     printf("Please input send msg:\n");  
  49.         gets(buffer);  
  50.   
  51.         n_write = write(sockfd,buffer,strlen(buffer));  
  52.   
  53.     if(n_write == -1)  
  54.     {  
  55.         perror("send to server msg error!\n");  
  56.         exit(1);  
  57.     }  
  58.     }  
  59.       
  60.     return 0;  
  61. }  

并发服务器模型的缺点是:客户端不再和服务器交互时,其多线程仍在工作,比较耗费CPU 的资源。

I/O多路转接模型:

利用一个“监听者”,当有客户端发出连接请求及客户端发送信息时,才会和服务器连接。

服务器代码:

[cpp]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. #include <stdlib.h>  
  2. #include <stdio.h>  
  3. #include <errno.h>  
  4. #include <string.h>  
  5. #include <netdb.h>  
  6. #include <sys/types.h>  
  7. #include <netinet/in.h>  
  8. #include <sys/socket.h>  
  9. #include<unistd.h>  
  10. #include<arpa/inet.h>  
  11. #include<ctype.h>  
  12.   
  13.   
  14. #define portnumber 8000  
  15.   
  16. #define MAX_LINE 80  
  17.   
  18.   
  19.   
  20. int main(void)  
  21. {  
  22.     int  lfd;  
  23.     int cfd;  
  24.     int sfd;  
  25.     int rdy;  
  26.   
  27.     struct sockaddr_in sin;  
  28.     struct sockaddr_in cin;  
  29.   
  30.     int client[FD_SETSIZE];    
  31.   
  32.     int maxi;  
  33.     int maxfd;                         
  34.   
  35.     fd_set rset;  
  36.     fd_set allset;  
  37.   
  38.     socklen_t addr_len;           
  39.   
  40.     char buffer[MAX_LINE];  
  41.   
  42.     int i;  
  43.     int n;  
  44.     int len;  
  45.     int opt = 1;     
  46.   
  47.     char addr_p[20];  
  48.   
  49.   
  50.     bzero(&sin,sizeof(struct sockaddr_in));     
  51.     sin.sin_family=AF_INET;                   
  52.     sin.sin_addr.s_addr=htonl(INADDR_ANY);   
  53.     sin.sin_port=htons(portnumber);           
  54.   
  55.       
  56.     if((lfd=socket(AF_INET,SOCK_STREAM,0))==-1)   
  57.     {  
  58.         fprintf(stderr,"Socket error:%s\n\a",strerror(errno));  
  59.         exit(1);  
  60.     }  
  61.     printf("socket!\n");  
[cpp]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. setsockopt(lfd, SOL_SOCKET, SO_REUSEADDR, &opt, sizeof(opt));  
  2.   
  3.   
  4. if(bind(lfd,(struct sockaddr *)(&sin),sizeof(struct sockaddr))==-1)  
  5. {  
  6.     fprintf(stderr,"Bind error:%s\n\a",strerror(errno));  
  7.     exit(1);  
  8. }  
  9.    printf("bind!\n");  
  10.   
  11.   
  12. if(listen(lfd,20)==-1)  
  13. {  
  14.     fprintf(stderr,"Listen error:%s\n\a",strerror(errno));  
  15.     exit(1);  
  16. }  
  17.    printf("listen!\n");  
  18. printf("Accepting connections .......\n");  
  19.   
  20. maxfd = lfd;                                 
  21. maxi = -1;  
  22.   
  23.   
  24. for(i = 0;i < FD_SETSIZE;i++)  
  25. {  
  26.        client[i] = -1;  
  27. }  
  28. FD_ZERO(&allset);                       
  29. FD_SET(lfd,&allset);                      
  30.   
  31.   
  32.   
  33. while(1)  
  34. {  
  35.     rset = allset;  
  36.        printf("selecting!\n");  
  37.   
  38.     rdy = select(maxfd + 1, &rset, NULL, NULL, NULL);  
  39.   
  40.     printf("selected!\n");  
  41.     if(FD_ISSET(lfd, &rset))  
  42.     {  
  43.   
  44.         addr_len = sizeof(sin);  
  45.                        printf("accepting!\n");  
  46.           
  47.         if((cfd=accept(lfd,(struct sockaddr *)(&cin),&addr_len))==-1)  
  48.         {  
  49.             fprintf(stderr,"Accept error:%s\n\a",strerror(errno));  
  50.             exit(1);  
  51.         }  
  52.         printf("accepted!\n");  
  53.   
  54.   
  55.                       
  56.         for(i = 0; i<FD_SETSIZE; i++)  
  57.         {       //printf("%d\t",client[i]);  
  58.             if(client[i] <= 0)  
  59.             {  
  60.                 client[i] = cfd;     
  61.                 break;  
  62.             }  
  63.         }  
  64.   
  65.       
  66.         if(i == FD_SETSIZE)  
  67.         {  
  68.             printf("too many clients");  
  69.             exit(1);  
  70.         }  
  71.   
  72.         FD_SET(cfd, &allset);       
  73.   
  74.         if(cfd > maxfd)                    
  75.         {  
  76.             maxfd = cfd;  
  77.         }  
  78.   
  79.         if(i > maxi)  
  80.         {  
  81.             maxi = i;  
  82.         }  
  83.   
  84.         if(--rdy <= 0)                 
  85.         {  
  86.             continue;  
  87.         }  
  88.   
  89.     }  
  90.       
  91.     for(i = 0;i< FD_SETSIZE;i++)  
  92.     {  
  93.         if((sfd = client[i]) < 0)  
  94.         {  
  95.             continue;  
  96.         }  
  97.   
  98.         if(FD_ISSET(sfd, &rset))  
  99.         {  
  100.             printf("reading!\n");  
  101.             n = read(sfd,buffer,MAX_LINE);  
  102.             printf("%s\n",buffer);  
  103.             printf("read!\n");  
  104.             if(n == 0)  
  105.             {  
  106.                 printf("the other side has been closed. \n");  
  107.                 fflush(stdout);                                      
  108.                 close(sfd);  
  109.   
  110.                 FD_CLR(sfd, &allset);                         
  111.                 client[i] = -1;  
  112.             }  
  113.   
  114.             else  
  115.             {  
  116.                   
  117.                 inet_ntop(AF_INET, &cin.sin_addr, addr_p, sizeof(addr_p));  
  118.                 addr_p[strlen(addr_p)] = '\0';  
  119.   
  120.                   
  121.                 printf("Client Ip is %s, port is %d\n",addr_p,ntohs(cin.sin_port));  
  122.   
  123.   
  124.               
  125.                 if(n == 1)  
  126.                 {  
  127.                     exit(1);  
  128.                 }  
  129.             }  
  130.   
  131.               
  132.             if(--rdy <= 0)  
  133.             {  
  134.                 break;  
  135.               
  136.   
  137.         }  
  138.     }  
  139.   
  140. }  
  141.   
  142. close(lfd);         
  143. return 0;  
多路转接客户端:

[cpp]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. #include <stdlib.h>   
  2. #include <stdio.h>   
  3. #include <errno.h>   
  4. #include <string.h>   
  5. #include <netdb.h>   
  6. #include <sys/types.h>   
  7. #include <netinet/in.h>   
  8. #include <sys/socket.h>   
  9.   
  10. #define portnumber 8000  
  11.   
  12. int main(int argc, char *argv[])   
  13. {   
  14.     int nbytes;  
  15.     int sockfd;   
  16.     char buffer[80];   
  17.     char buffer_2[80];  
  18.     struct sockaddr_in server_addr;   
  19.     struct hostent *host;   
  20.   
  21.     if(argc!=2)   
  22.     {   
  23.         fprintf(stderr,"Usage:%s hostname \a\n",argv[0]);   
  24.         exit(1);   
  25.     }   
  26.   
  27.     if((host=gethostbyname(argv[1]))==NULL)   
  28.     {   
  29.         fprintf(stderr,"Gethostname error\n");   
  30.         exit(1);   
  31.     }   
  32.   
  33.        
  34.     if((sockfd=socket(AF_INET,SOCK_STREAM,0))==-1) // AF_INET:Internet;SOCK_STREAM:TCP  
  35.     {   
  36.         fprintf(stderr,"Socket Error:%s\a\n",strerror(errno));   
  37.         exit(1);   
  38.     }   
  39.   
  40.       
  41.     bzero(&server_addr,sizeof(server_addr)); //   
  42.     server_addr.sin_family=AF_INET;  
  43.     server_addr.sin_port=htons(portnumber);  
  44.     server_addr.sin_addr = *((struct in_addr *)host->h_addr);//址  
  45.   
  46.       
  47.     if(connect(sockfd,(struct sockaddr *)(&server_addr),sizeof(struct sockaddr))==-1)   
  48.     {   
  49.         fprintf(stderr,"Connect Error:%s\a\n",strerror(errno));   
  50.         exit(1);   
  51.     }   
  52.   
  53.     while(1){  
  54.       
  55.     printf("Please input char:\n");  
  56.       
  57.       
  58.     fgets(buffer,1024,stdin);   
  59.     write(sockfd,buffer,strlen(buffer));  
  60.        #if 0  
  61.     if((nbytes=read(sockfd,buffer_2,81))==-1)   
  62.         {   
  63.             fprintf(stderr,"Read Error:%s\n",strerror(errno));  
  64.             exit(1);   
  65.         }  
  66.           
  67.         buffer_2[nbytes]='\0';  
  68.         printf("Client received from Server %s\n",buffer_2);  
  69.         #endif  
  70.       
  71.   
  72.     }     
  73.     close(sockfd);  
  74.     exit(0);   
  75. }   
  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值