首页 > 编程语言 >UNIX网络编程:socket & select() 实现clients/server通信

UNIX网络编程:socket & select() 实现clients/server通信

时间:2023-05-26 11:22:05浏览次数:43  
标签:socket int clients server client connfd include conn

一、问题引入

UNIX网络编程 卷1:套接字联网API(第三版) 第6章 介绍了I/O复用可以通过select()的单进程服务器与多客户端通信。

UNIX下可用的5中I/O模型:

  • 阻塞式I/O
  • 非阻塞式I/O
  • I/O(select和poll)
  • 信号驱动式I/O(SIGIO)
  • 异步I/O(POSIX的aio_系列函数)

其中前面4种可以分为同步I/O,第五种为异步I/O。

二、解决过程

client 代码无需修改,请参考 Linux网络编程:socket & fork()多进程 实现clients/server通信

2-1 server 代码

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

#define MYPORT 8887
#define BACKLOG 5
#define BUF_SIZE 1024

typedef struct CLIENT_CONN_ST
{
    int connfd;
    char socket[128];
    struct sockaddr_in cliaddr;
} CLIENT_CONN_ST;

static CLIENT_CONN_ST g_client_conn[BACKLOG];

static void client_connect_index_init(void)
{
    for (int i = 0; i < BACKLOG; i++)
    {
        memset(&g_client_conn[i], 0, sizeof(struct CLIENT_CONN_ST));
        g_client_conn[i].connfd = -1;
    }
}

static int client_connect_index_find(void)
{
    int i;
    for (i = 0; i < BACKLOG; i++)
    {
        if (g_client_conn[i].connfd == -1)
            break;
    }
    return i;
}

static int client_connect_close(void)
{
    for (int i = 0; i < BACKLOG; i++)
    {
        if (g_client_conn[i].connfd != -1)
        {
            close(g_client_conn[i].connfd);
            g_client_conn[i].connfd = -1;
        }
    }
}

static int string_toupper(const char *src, int str_len, char *dst)
{
    int i;
    for (i = 0; i < str_len; i++)
    {
        dst[i] = toupper(src[i]);
    }
    return i;
}

int main(void)
{
    int listenfd, connfd;
    struct sockaddr_in server_addr;
    struct sockaddr_in client_addr;
    socklen_t addr_len;
    int opt = 1;
    char buf[BUF_SIZE];
    int recv_len, send_len;
    char read_buf[BUF_SIZE], write_buf[BUF_SIZE];
    int ret;
    int idx;

    fd_set fdsr;
    int maxsock;
    struct timeval tv;

    server_addr.sin_family = AF_INET;
    server_addr.sin_port = htons(MYPORT);
    server_addr.sin_addr.s_addr = INADDR_ANY;
    if ((listenfd = socket(AF_INET, SOCK_STREAM, 0)) < 0)
    {
        perror("socket");
        exit(EXIT_FAILURE);
    }
    // 端口复用
    if (setsockopt(listenfd, SOL_SOCKET, SO_REUSEADDR, &opt, sizeof(opt)) < 0)
    {
        perror("setsockopt");
        exit(EXIT_FAILURE);
    }
    if (bind(listenfd, (struct sockaddr *)&server_addr, sizeof(server_addr)) < 0)
    {
        perror("bind");
        exit(EXIT_FAILURE);
    }
    if (listen(listenfd, BACKLOG) < 0)
    {
        perror("listen");
        exit(EXIT_FAILURE);
    }
    printf("listen port %d\n", MYPORT);
    maxsock = listenfd;
    client_connect_index_init();
    while (1)
    {
        // initialize file descriptor set
        FD_ZERO(&fdsr);
        FD_SET(listenfd, &fdsr);
        // initialize timeout val
        tv.tv_sec = 30;
        tv.tv_usec = 0;
        // add active connection to fd set
        for (int i = 0; i < BACKLOG; i++)
        {
            if (g_client_conn[i].connfd != -1)
                FD_SET(g_client_conn[i].connfd, &fdsr);
        }
        ret = select(maxsock + 1, &fdsr, NULL, NULL, &tv);
        if (ret < 0)
        {
            perror("select");
            break;
        }
        else if (ret == 0)
        {
            printf("30s timeout\n");
            continue;
        }
        // check every fd in the set
        for (int i = 0; i < BACKLOG; i++)
        {
            if (FD_ISSET(g_client_conn[i].connfd, &fdsr))
            {
                memset(read_buf, 0, sizeof(read_buf));
                memset(write_buf, 0, sizeof(write_buf));
                recv_len = read(g_client_conn[i].connfd, read_buf, sizeof(read_buf));
                if (recv_len <= 0) // client close
                {
                    printf("%s close\n", g_client_conn[i].socket);
                    close(g_client_conn[i].connfd);
                    FD_CLR(g_client_conn[i].connfd, &fdsr);
                    g_client_conn[i].connfd = -1;
                }
                else
                {
                    printf("%s:%s(%d Byte)\n", g_client_conn[i].socket, read_buf, recv_len);
                    if (strcmp("exit", read_buf) == 0)
                    {
                        printf("%s exit\n", g_client_conn[i].socket);
                        close(g_client_conn[i].connfd);
                        FD_CLR(g_client_conn[i].connfd, &fdsr);
                        g_client_conn[i].connfd = -1;
                    }
                    else
                    {
                        send_len = string_toupper(read_buf, strlen(read_buf), write_buf);
                        write(g_client_conn[i].connfd, write_buf, send_len);
                    }
                }
            }
        }
        // check whether a new connection comes
        if (FD_ISSET(listenfd, &fdsr))
        {
            memset(&client_addr, 0, sizeof(client_addr));
            addr_len = sizeof(client_addr);
            connfd = accept(listenfd, (struct sockaddr *)&client_addr, &addr_len);
            if (connfd <= 0)
            {
                perror("accept");
                continue;
            }
            idx = client_connect_index_find();
            if (idx == BACKLOG)
            {
                printf("client connected upper limit, refused connect\n");
                close(connfd);
                continue;
            }
            // add to fd queue
            g_client_conn[idx].connfd = connfd;
            memset(&g_client_conn[idx].socket, 0, sizeof(g_client_conn[idx].socket));
            printf("client addr:%s por:%d\n",
                   inet_ntop(AF_INET, &client_addr.sin_addr, buf, sizeof(buf)),
                   ntohs(client_addr.sin_port));
            snprintf(g_client_conn[idx].socket, sizeof(g_client_conn[idx].socket), "client socket (%s:%d)",
                     inet_ntop(AF_INET, &client_addr.sin_addr, buf, sizeof(buf)),
                     ntohs(client_addr.sin_port));
            if (connfd > maxsock)
                maxsock = connfd;
        }
    }
    // close all connections
    client_connect_close();

    exit(EXIT_SUCCESS);
}

2-2 编译运行结果

  • 编译server.c

gcc server.c -g -std=gnu99 -o server

  • 客户端连接服务器

  • 客户端与服务器通信

三、反思总结

通过select()的I/O复用可以实现多客户端通信。

ret = select(maxsock + 1, &fdsr, NULL, NULL, &tv); 完成了对新客户端连接和已连接客户端的监控,可以在循环中检查判断I/O的状态。

由于是多客户端程序,故需要一个for语句来判断每一个已连接套接字的I/O状态。

for (int i = 0; i < BACKLOG; i++)
{
    if (FD_ISSET(g_client_conn[i].connfd, &fdsr))
    {
        ;
    }

标签:socket,int,clients,server,client,connfd,include,conn
From: https://www.cnblogs.com/caojun97/p/17420899.html

相关文章

  • sqlserver 游标的使用
    @@SQLserver游标 游标游标可以对一个select的结果集进行处理,或是不需要全部处理,就会返回一个对记录集进行处理之后的结果。1、游标实际上是一种能从多条数据记录的结果集中每次提取一条记录的机制。游标可以完成:#允许定位到结果集中的特定行#从结果集的当前位置检索一行或......
  • windows server2016 操作系统修改默认远程端口
    一、需求   远程端口,windows默认的3389.linux的22,这种都是知名端口,如果IP地址暴露,很可能会被攻击,这时候就需要更改端口号。二、操作步骤2.1打开注册表   快捷键WIN+R,命令行窗口输入regedit2.2进入以下路径  这里是默认端口,修改为自己除1024以后,以及未被......
  • Python 数据库Insert语句脚本生成工具(SQL Server)
    编写这个小工具,是因为平时部署项目的时候,需要导出一些公共的数据(权限、参数设置等),覆盖插入正式环境。话不多说,直接上代码:importpyodbcimportwarningsimportdecimalimportwinregimportosimportconfigparserimporttimeimportdatetimewarnings.filterwarnings('igno......
  • 【转载】Sqlserver存储过程中使用Select和Set给变量赋值
    @@sqlserverselect赋值  Sqlserver存储过程是时常使用到的一个数据库对象,在存储过程中会使用到Declare来定义存储过程变量,定义的存储过程变量可以通过Set或者Select等关键字方法来进行赋值操作,使用Set对存储过程变量赋值为直接赋值,使用Select则一般从数据表中查找出符合条......
  • 【转】SQL SERVER 游标
    @@SQLserver游标  游标(Cursor)是SQL Server的一种数据访问机制,它使得程序可以逐行处理数据,即允许用户访问单独的数据行,对每一行数据进行单独的处理。一,创建游标对象创建游标对象,注意不是游标变量,游标名称不需要带前导@:DECLAREcursor_nameCURSOR[LOCAL|GLO......
  • SqlServer中使用Select语句给变量赋值的时候需要注意的一个问题
    @@sqlserverselect赋值  我们知道在SqlServer中可以用Select语句给变量赋值,比如如下语句就为int类型的变量@id赋值1declare@idint=-1;23select@id=idfrom4(5select1asid6unionall7select2asid8unionall9select3asid10)a......
  • SqlServer select 赋值问题
    @@sqlserverselect赋值 --变量赋值正确,单个数据默认为变量declare@s1varchar(20)set@s1=(selectMAX(parked_id)fromparked)--变量赋值错误,多个数据默认为数据集declare@s2varchar(20)set@s2=(selectparked_idfromparked)————————————————版权......
  • 【RocketMQ】NameServer总结
    NameServer是一个注册中心,提供服务注册和服务发现的功能。NameServer可以集群部署,集群中每个节点都是对等的关系(没有像ZooKeeper那样在集群中选举出一个Master节点),节点之间互不通信。服务注册Broker启动的时候会向所有的NameServer节点进行注册,注意这里是向集群中所有的NameServ......
  • sql server 中游标详解
    @@SQLserver游标 目录游标的定义种类:(我也不太理解,有理解的@我)游标的类型:游标的实现游标的实现功能游标的使用的步骤:游标的使用:1.声明游标游标的格式:(看了很多版本的格式说明,这是最好理解的)声明一个动态游标:2.打开游标3读取数据读取数据的格式:读取......
  • SqlServer——临时表
    @@sqlserver临时表 SqlServer——临时表 1、表的类型:SqlServer数据库中分为两个表:永久表、临时表;通过表名的前缀区分。永久表:与物理文件、C#中的静态类 类似,任何用户均可对其执行操作并且相互影响;临时表:简单的说就是使用时创建,断开连接即自动drop。2、临时表......