• TCP回射客户服务器模型(02 设置套接字选项、处理多并发)


      int setsockopt(int sockfd, int level, int optname, const void *optval, socklen_t optlen);  //设置套接字选项


      回射服务器编写,服务器关闭后处于TIME_WAIT状态,过一段时间才可以绑定刚才的端口。
    在绑定服务器之前尽可能调用setsockopt来设置REUSEADDR套接字选项,使用REUSEADDR选项可以使得不必等待TIME_WAIT状态消失就可以重启服务器
      服务器处理多并发,客户端短连接过来就创建子进程通信,客户端关闭后,子进程要退出.父进程一直处于监听。
      

      服务器进程程序:

    #include<unistd.h>
    #include<sys/types.h>
    #include<sys/socket.h>
    #include<string.h>
    #include<stdlib.h>
    #include<stdio.h>
    #include<errno.h>
    #include<netinet/in.h>
    #include<arpa/inet.h>
    #define ERR_EXIT(m)
        do
        {
            perror(m);
            exit(EXIT_FAILURE);
        }while(0)
    void do_service(int conn)
    {
            char recvbuf[1024];
            while(1)
            {
                memset(recvbuf,0,sizeof(recvbuf));
                int ret=read(conn,recvbuf,sizeof(recvbuf));
                //捕捉客户端关闭            
                if(ret==0)
                {
                    printf("client close
    ");
                    break;//不用继续循环等待客户端数据
                }
                else if(ret==-1)
                    ERR_EXIT("read error");
                fputs(recvbuf,stdout);
                write(conn,recvbuf,ret);
            }
    }
    int main(void)
    {
        int listenfd;
        if((listenfd=socket(PF_INET,SOCK_STREAM,IPPROTO_TCP))<0)
            ERR_EXIT("socket error");
        //if((listenfd=socket(PF_INET,SOCK_STREAM,0))<0)
        
    
        //本地协议地址赋给一个套接字
        struct sockaddr_in servaddr;
        memset(&servaddr,0,sizeof(servaddr));
        servaddr.sin_family=AF_INET;
        servaddr.sin_port=htons(5188);
        servaddr.sin_addr.s_addr=htonl(INADDR_ANY);//表示本机地址
        //servaddr.sin_addr.s_addr=inet_addr("127.0.0.1");
        //inet_aton("127.0.0.1",&servaddr.sin_addr);
        
        //绑定之前,开启地址重复使用,关闭服务器再打开不用等待TIME_WAIT
        int on=1;
        if(setsockopt(listenfd,SOL_SOCKET,SO_REUSEADDR,&on,sizeof(on))<0)
            ERR_EXIT("setsockopt error");
        //绑定本地套接字
        if(bind(listenfd,(struct sockaddr*)&servaddr,sizeof(servaddr))<0)
            ERR_EXIT("bind error");
        if(listen(listenfd,SOMAXCONN)<0)//设置监听套接字(被动套接字)
            ERR_EXIT("listen error");
        
        struct sockaddr_in peeraddr;//对方套接字地址
        socklen_t peerlen=sizeof(peeraddr);/一定要初始化
        int conn;//已连接套接字(主动套接字)
        pid_t pid;
      //原先的程序为何不能处理多客户端连接:因为之前的服务器程序,一旦客户端请求连接,服务器就会一直处于while循环,处理客户端请求,不再处理accept
    while(1){ if((conn=accept(listenfd,(struct sockaddr*)&peeraddr,&peerlen))<0) ERR_EXIT("accept error");//accept本质是到已连接队列中,取第一个连接。 //连接好之后就构成连接,端口是客户端的。peeraddr是对端 printf("ip=%s port=%d ",inet_ntoa(peeraddr.sin_addr),ntohs(peeraddr.sin_port)); pid=fork(); if(pid==-1) ERR_EXIT("fork"); if(pid==0){ close(listenfd);//父子进程共享描述符 do_service(conn); //do_service返回,某个客户端关闭,结束该子进程,否则子进程也去接受连接(accept)。 exit(EXIT_SUCCESS); }else close(conn); } return 0; }

      客户端程序:

     1 #include<unistd.h>
     2 #include<sys/types.h>
     3 #include<sys/socket.h>
     4 #include<string.h>
     5 #include<stdlib.h>
     6 #include<stdio.h>
     7 #include<errno.h>
     8 #include<netinet/in.h>
     9 #include<arpa/inet.h>
    10 #define ERR_EXIT(m)
    11     do
    12     {
    13         perror(m);
    14         exit(EXIT_FAILURE);
    15     }while(0)
    16 int main(void)
    17 {
    18     int sock;//客户端创建套接字
    19     if((sock=socket(PF_INET,SOCK_STREAM,IPPROTO_TCP))<0)
    20         ERR_EXIT("socket error");
    21     
    22     struct sockaddr_in servaddr;//本地协议地址赋给一个套接字
    23     memset(&servaddr,0,sizeof(servaddr));
    24     servaddr.sin_family=AF_INET;
    25     servaddr.sin_port=htons(5188);
    26     
    27     servaddr.sin_addr.s_addr=inet_addr("127.0.0.1");//服务器段地址
    28     //inet_aton("127.0.0.1",&servaddr.sin_addr);
    29     
    30     if(connect(sock,(struct sockaddr*)&servaddr,sizeof(servaddr))<0)
    31         ERR_EXIT("connect");
    32     char sendbuf[1024]={0};
    33     char recvbuf[1024]={0};
    34     while(fgets(sendbuf,sizeof(sendbuf),stdin)!=NULL)//默认有换行符
    35     {
    36         write(sock,sendbuf,strlen(sendbuf));
    37         read(sock,recvbuf,sizeof(recvbuf));
    38         fputs(recvbuf,stdout);
    39 
    40         memset(sendbuf,0,sizeof(sendbuf));
    41         memset(recvbuf,0,sizeof(recvbuf));
    42     }
    43     close(sock);
    44     
    45     return 0;
    46 }
  • 相关阅读:
    从零开始整SpringBoot-工具与插件
    算法与数据结构学习笔记(目录)
    牛客小白月赛30(个人题解)
    Manjaro 上手使用简明教程
    C++函数:std::tie 详解
    Educational Codeforces Round 99 (Rated for Div. 2) (A ~ F)个人题解
    VS Code C++ 项目快速配置模板
    【字符串算法】字典树详解
    关于算法竞赛某些常见基础错误
    Teacher Ma专场
  • 原文地址:https://www.cnblogs.com/wsw-seu/p/8409926.html
Copyright © 2020-2023  润新知