TCP回射服务器修订版(ubuntu 18.04)

一、需求

  把https://www.cnblogs.com/soldierback/p/10673345.html中的TCP回射服务器程序重写成使用select来处理任意个客户的单进程

  程序,而不是为每个进程派生一个子进程

二、分析

 (1)服务器有单个监听描述符

     

 (2)服务器只维护一个读描述符集;假设服务器是在前台启动的,那么描述符0、1、2将分别被设置为标准输入、标准输出和标准错误输出;可见监听

     套接字的第一个可用描述符是3

     

 (3)服务器维护一个名为clients的整型数组,它包含每个客户的已连接套接字描述符,该数组的所有元素都被初始化为-1

        

 (4)当第一个客户与服务器建立连接时,监听描述符变为可读,服务器于是调用accept

    

 (5)假设由accept返回的描述符为4,则clients数组和读描述符集如下所示

        

 (6)当第二个客户与服务器建立连接时,假设由accept返回的描述符为5,则clients和都描述符集如下所示

     

 (7)假设第一个客户终止它的连接;该客户的TCP发送一个FIN,使得服务器的描述符4变为可读;当服务器读这个已连接套接字时,read将

     返回0,服务器于是关闭该套接字并相应地更新数据结构:把clients[0]的值置为-1,把描述符集中描述符4的位设置为0;注意:maxfd的

     值没有改变

     

三、源代码

#include <stdio.h>
#include <unistd.h>
#include <arpa/inet.h>
#include <sys/socket.h>
#include <strings.h>

#define LISTENQ 1024
#define MAXLINE 4096
#define SERV_PORT 9999
#define SA struct sockaddr

ssize_t writen(int, const void*, size_t);
char *sock_ntop(const struct sockaddr*, socklen_t);

int main(int argc, char *argv[]) {

    int i, maxi, maxfd, listenfd, connfd, sockfd;
    int nready, clients[FD_SETSIZE];
    ssize_t n;
    fd_set rset, allset;
    char buf[MAXLINE];
    socklen_t clilen;
    struct sockaddr_in cliaddr, servaddr;

    listenfd = socket(AF_INET, SOCK_STREAM, 0);

    bzero(&servaddr, sizeof(servaddr));
    servaddr.sin_family      = AF_INET;
    servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
    servaddr.sin_port        = htons(SERV_PORT);

    bind(listenfd, (SA *) &servaddr, sizeof(servaddr));

    listen(listenfd, LISTENQ);

    maxfd = listenfd;     /* initialize */
    maxi = -1;            /* index into clients[] array */
    for (i = 0; i < FD_SETSIZE; i++) {
        clients[i] = -1;  /* -1 indicates available entry */
    }
    FD_ZERO(&allset);
    FD_SET(listenfd, &allset);

    for ( ; ; ) {
        rset = allset;             /* structure assignment */
        nready = select(maxfd+1, &rset, NULL, NULL, NULL);

        if (FD_ISSET(listenfd, &rset)) { /* new client connection */
            clilen = sizeof(cliaddr);
            connfd = accept(listenfd, (SA *) &cliaddr, &clilen);

            for (i = 0; i < FD_SETSIZE; i++) {
                if (clients[i] < 0) {
                    clients[i] = connfd; /* save descriptor */
                    printf("new client %d: [%s]\n", i, sock_ntop((SA *)&cliaddr, sizeof(cliaddr)));
                    break;
                }
            }
            if (i == FD_SETSIZE) {
                printf("too many clients\n");
                goto ifAnyDescriptorReadable;
            }

            FD_SET(connfd, &allset);     /* add new descriptor to set */
            if (connfd > maxfd)
                maxfd = connfd;          /* for select */
            if (i > maxi)
                maxi = i;                /* max index in client[] array */

            ifAnyDescriptorReadable:
            if (--nready <= 0)
                continue;                /* no more readable descriptors */
        }

        for (i = 0; i <= maxi; i++) {    /* check all clients for data */
            if ( (sockfd = clients[i]) < 0) {
                continue;
            }
            if (FD_ISSET(sockfd, &rset)) {
                if ( (n = read(sockfd, buf, MAXLINE)) == 0) {
                                         /* connection closed by client */
                    close(sockfd);
                    FD_CLR(sockfd, &allset);
                    clients[i] = -1;
                    printf("client [%d] quit\n", i);
                } else {
                    writen(sockfd, buf, n);
                }

                if (--nready <= 0) {
                    break;               /* no more readable descriptors */
                }
            }
        }
    }
}

注:sock_ntop和writen两个函数在分类为《UNIX网络编程》的其他随笔中有存在的问题:某个客户建立连接后不断发送数据,此时会导致服务器拒绝为其他客户服务解决方法:让每个客户由单独的进程或线程提供服务

原文地址:https://www.cnblogs.com/soldierback/p/10704122.html

时间: 2024-12-25 23:58:59

TCP回射服务器修订版(ubuntu 18.04)的相关文章

TCP回射服务器/客户端分析

本文将对一个简单的TCP回射服务器和客户端进行抓包,从而分析一次成功而理想TCP会话的基本流程,多次不成功或与预期不一致的抓包结果将在下篇博文进行分析 本文程序编译环境为: Linux version 3.16.4-1-ARCH gcc version 4.9.1 20140903 (prerelease) Glibc 2.18 服务器代码如下: 1 #include <unistd.h> 2 #include <sys/types.h> 3 #include <sys/so

TCP回射服务器程序:main函数

TCP回射并发服务器 1.创建套接字,绑定服务器的众所周知端口 创建一个TCP套接字,在待绑定到该TCP套接字的网际网套接字地址结构中填入通配地址(INADDR_ANY) 和服务器的众所知周(SERV_PORT,在头文件中unp.h中其定义为9877) 绑定通配地址是在告知系统: 要是系统是多宿主机,我们将接受目的地地址为任何本地接口的连接 我们对TCP端口号的选择应该比1023大,比5000大,比49152小,而且不和任何注册的端口冲突 listen把该套接字地址转换成一个监听套接字 2.等待

TCP回射服务器程序:str_echo函数

str_echo函数执行处理每个客户的服务: 从客户读入数据,并把它们回射给客户 读入缓冲区并回射其中内容: read函数从套接字读入数据,writen函数把其中内容回射给客户 如果客户关闭连接,那么接收到客户的FIN将导致服务器子进程的read函数返回0,这又导致str_echo函数的返回,从而终止子进程 #include "unp.h" void str_echo(int sockfd) { ssize_t n; char buf[MAXLINE]; again: while (

UNIX网络编程卷1 回射服务器程序 TCP服务器程序设计范式 四个版本

本文为senlie原创,转载请保留此地址:http://blog.csdn.net/zhengsenlie 这是一个简单的回射服务器程序.它将客户发送的数据读入缓冲区并回射其中内容 下面我会介绍同一个使用 TCP 协议的回射服务器程序的几个不同版本,分别是 fork 版本.select 版本.poll 版本.多线程版本 fork 版本:为每一个客户连接派生(fork) 一个子进程用来处理客户请求 /** * TCP/IPv4 协议相关 * **/ #include "unp.h" in

第九章 TCP和UDP同时用复用一个端口实现一个回射服务器

#include <sys/types.h> #include <sys/socket.h> #include <netinet/in.h> #include <arpa/inet.h> #include <assert.h> #include <stdio.h> #include <unistd.h> #include <errno.h> #include <string.h> #include

Ubuntu 18.04安装Samba服务器及配置

Ubuntu 18.04安装Samba服务器及配置 局域网下使用samba服务在Linux系统与Windows系统直接共享文件是一项很方便的操作.以Ubuntu为例配置samba服务,Linux服务器的版本是Ubuntu 18.04.1 LTS. 在终端中执行下列指令:cat /etc/issue查看当前正在运行的 Ubuntu 的版本号. 以下是我的安装配置步骤: (1)更新当前软件. sudo apt-get upgrade  sudo apt-get update  sudo apt-ge

Ubuntu 18.04配置邮箱服务器(局域网)

-------------------------------------------Ubuntu 18.04配置邮箱服务器(局域网)-------------------------------------------一.安装 Ubuntu 18.04附带了systemd-resolve,需要禁用它,因为它绑定到53端口,与Dnsmasq端口冲突.  运行以下命令以禁用已解析的服务: [email protected]:$ systemctl disable systemd-resolved

第十二篇:并发回射服务器的具体实现及其中僵尸子进程的清理( 上 )

前言 本文将分为两个部分,第一部分具体实现一对并发回射服务器/客户程序( 看过前面那篇文章的这部分可不看 重复了 ):第二部分为服务器添加僵尸子进程自动清理机制. 那么服务器具体怎么实现并发?怎么会有僵尸进程?僵尸进程又是什么?如何处理这些僵尸进程 ... 本文将为你一一解惑. 回射并发服务器 功能:接收用户发送过来的数据后再发送回用户,且能同时处理多个用户请求. 大体思路:每当收到用户请求,服务器就fork一个子进程,让子进程去处理客户请求. 实现代码: 1 #include "unp.h&q

Ubuntu 18.04修改ip步骤

Ubuntu 18.04修改ip步骤: 注:配置/etc/network/interfaces已无用 vi /etc/netplan/50-cloud-init.yaml network:ethernets:eno1:addresses: [10.0.0.2/24]dhcp4: nogateway4: 10.0.0.1 :wq 注:1.":"后要有空格2.Ubuntu 18.04的网卡名已改为eno1.eno2.... 使配置生效(不需重启服务器): netplan apply 原文地