IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 系统运维 -> 实现TCP并发服务器---IO多路复用 -> 正文阅读

[系统运维]实现TCP并发服务器---IO多路复用

实现TCP并发服务器—IO多路复用

1.服务器模型

1.1 概念

服务器模型主要分为两种,循环服务器,并发服务器

循环服务器:服务器在同一时间只能处理一个客户端的请求。

并发服务器:服务器在同一时间内能同时处理多个客户端的请求。

TCP的服务器默认的就是一个循环服务器 有两个阻塞函数( accept recv) 会相互影响

UDP的服务器默认的就是一个并发服务器,因为只有一个阻塞函数( recvfrom)

1.2 TCP并发服务器

有些场景下,我们既要保证数据可靠,又要支持并发,这就需要用到TCP并发服务器

如何实现TCP并发服务器?

? (1)使用多进程实现TCP并发服务器

? (2)使用多线程实现TCP并发服务器

? (3)使用多路IO复用实现TCP并发服务器

对于实际开发过程中,使用多进程实现TCP并发服务器,并发量大的时候,对系统的资源占用量也会很大

如果使用多线程,业务逻辑复杂的时候,又涉及到临近资源访问的问题

比较好的方式是使用多路IO复用实现TCP并发服务器

2.select实现TCP并发

2.1select函数说明

功能:
	实现多路IO复用
头文件:
	#include <sys/select.h>
函数原型:
	int select(int nfds, fd_set *readfds, fd_set *writefds,
                fd_set *exceptfds, struct timeval *timeout);
参数:
	nfds:     监视的最大的文件描述符 +1
	readfds:  要监视的读文件描述符集合,如果不关心,可以传NULL
	writefds: 要监视的写文件描述符集合,如果不关心,可以传NULL
	exceptfds:要监视的异常的文件描述符集合,如果不关心,可以传NULL
				(一般我们只关心readfds)
	timeout:超时时间
         0  非阻塞
         NULL 永久阻塞
         结构体 阻塞一定的时间
     --暂时先不用管,后面网络超时检测再讲

返回值:
	成功 就绪的文件描述符的个数
	失败 -1
	超时 0
	
	void FD_CLR(int fd, fd_set *set);	//将文件描述符在集合中删除
	int  FD_ISSET(int fd, fd_set *set);	//判断文件描述符是否在集合中
										//0 不在里面了  非0 在里面
	void FD_SET(int fd, fd_set *set);	//将文件描述符添加到集合中
	void FD_ZERO(fd_set *set);			//清空集合

注意:
	1.select只能监视小于 FD_SETSIZE(1024) 的文件描述符 
 		本质是用数组(128字节)保存 每一个bit位监视一个fd
	2.select函数在返回时会将没有就绪的文件描述符在表中擦除
		所以,在循环中调用select时,每次需要重新填充集合

2.2select代码说明

#include <arpa/inet.h>
#include <netinet/in.h>
#include <netinet/ip.h>
#include <pthread.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/select.h>
#include <sys/socket.h>
#include <sys/types.h>
#include <unistd.h>

#define ERRLOG(msg)                                         \
    do {                                                    \
        printf("%s %s %d\n", __FILE__, __func__, __LINE__); \
        perror(msg);                                        \
        exit(-1);                                           \
    } while (0)

int main(int argc, const char* argv[])
{
    //入参合理性检查
    if (3 != argc) {
        printf("Usage: %s <Ip> <Port>", argv[0]);
        return -1;
    }
    //创建套接字
    int sockfd = 0;
    if (-1 == (sockfd = socket(AF_INET, SOCK_STREAM, 0))) {
        ERRLOG("socket error");
    }
	//填充网络信息结构体
    struct sockaddr_in serveraddr;
    serveraddr.sin_family = AF_INET;
    serveraddr.sin_addr.s_addr = inet_addr(argv[1]);
    serveraddr.sin_port = ntohs(atoi(argv[2]));

    socklen_t serveraddr_len = sizeof(serveraddr);
	//将网络信息结构体与套接字绑定
    if (-1 == bind(sockfd, (struct sockaddr*)&serveraddr, serveraddr_len)) {
        ERRLOG("bind error");
    }
    if (-1 == listen(sockfd, 5)) {
        ERRLOG("listen error");
    }
    struct sockaddr_in clientaddr;
    socklen_t clientaddr_len = sizeof(clientaddr);
    //创建要监视的文件描述符集合
    fd_set readfds;
    FD_ZERO(&readfds);
    fd_set readfds_temp;
    FD_ZERO(&readfds_temp);
    int max_fd = 0;
    //将文件描述符插入集合中
    FD_SET(sockfd, &readfds);
    //更新最大文件描述符集合
    max_fd = max_fd > sockfd ? max_fd : sockfd;

    int ret = 0;
    int i = 0;
    char buf[128] = { 0 };
    int nbytes = 0;
    while (1) {
        readfds_temp = readfds;
        int acceptfd = 0;
        if (-1 == (ret = select(max_fd + 1, &readfds_temp, NULL, NULL, NULL))) {
            ERRLOG("select error");
        } else {
            for (i = 3; i < max_fd + 1; i++) {
                if (FD_ISSET(i, &readfds_temp)) {
                    ret--;
                    //说明有文件描述符就位了
                    if (i == sockfd) {
                        //说明有客户端连接
                        if (-1 == (acceptfd = accept(sockfd, (struct sockaddr*)&clientaddr, &clientaddr_len))) {
                            ERRLOG("accept error");
                        }
                        printf("客户端[%d]连接到服务器...\n", acceptfd);
                        //将acceptfd放入集合中
                        FD_SET(acceptfd, &readfds);
                        max_fd > acceptfd ? max_fd : acceptfd;
                    } else {
                        //有客户端发来消息
                        memset(buf, 0, sizeof(buf));
                        if (-1 == (nbytes = recv(i, buf, sizeof(buf), 0))) {
                            ERRLOG("recv error");
                        } else if (0 == nbytes) {
                            printf("客户端[%d]断开了连接...\n", i);
                            close(i);
                            FD_CLR(i, &readfds);
                            continue;
                        }
                        if (!strcmp(buf, "quit")) {
                            printf("客户端[%d]退出了...\n", i);
                            close(i);
                            FD_CLR(i, &readfds);
                            continue;
                        }
                        printf("客户端[%d]发来数据[%s]\n", i, buf);
                        strcat(buf, "1115");
                        if (-1 == send(i, buf, sizeof(buf), 0)) {
                            ERRLOG("send error");
                        }
                    }
                }
            }
        }
    }
    close(sockfd);
    return 0;
}

3.poll实现TCP并发

3.1poll函数说明

功能:
	实现多路IO复用
头文件:
	#include <poll.h>
函数原型:
	int poll(struct pollfd *fds, nfds_t nfds, int timeout);
参数:
	fds:要监视的文件描述符集合的结构体数组的首地址
		使用的下面的结构体
			struct pollfd {
			   int   fd;         /* 文件描述符 */
			   short events;     /* 要监视的事件 */
			   short revents;    /* 返回的事件 */
			};
	
			events:使用位运算来或起来的 感兴趣的事件
					POLLIN	读事件 --我们一般只关注读
					POLLOUT	写事件
					POLLERR	异常事件
			revents:实际发生的事件
			
			//这种要监视的事件和实际发生的事件分开存储的方式
			//可以避免每次重置集合
		
	nfds:数组中实际有效的文件描述符的个数

	timeout:超时时间 单位毫秒
		-1		永久阻塞
		0		非阻塞
		5000	阻塞 5 秒
	
返回值:
	成功	就绪的文件描述符的个数
	失败	-1
	超时	0

3.2poll函数代码说明

#include <arpa/inet.h>
#include <netinet/in.h>
#include <netinet/ip.h>
#include <poll.h>
#include <pthread.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/socket.h>
#include <sys/types.h>
#include <unistd.h>

#define ERRLOG(msg)                                          \
    do {                                                     \
        printf("%s %s %d \n", __FILE__, __func__, __LINE__); \
        perror(msg);                                         \
        exit(-1);                                            \
    } while (0)

int main(int argc, const char* argv[])
{
    //入参合理性检查
    if (3 != argc) {
        printf("Usage:%s <Ip> <Port>", argv[0]);
        return -1;
    }
    //创建套接字
    int sockfd = 0;
    if (-1 == (sockfd = socket(AF_INET, SOCK_STREAM, 0))) {
        ERRLOG("socket error");
    }
    //填充服务器网络信息结构体
    struct sockaddr_in serveraddr;
    memset(&serveraddr, 0, sizeof(serveraddr));
    serveraddr.sin_family = AF_INET;
    serveraddr.sin_addr.s_addr = inet_addr(argv[1]);
    serveraddr.sin_port = htons(atoi(argv[2]));

    socklen_t serveraddr_len = sizeof(serveraddr);
    //将套接字与服务器的网络信息结构体绑定
    if (-1 == bind(sockfd, (struct sockaddr*)&serveraddr, serveraddr_len)) {
        ERRLOG("bind error");
    }
    //将套接字设置成被动监听状态
    if (-1 == listen(sockfd, 5)) {
        ERRLOG("listen error");
    }
    char buf[128] = { 0 };
    int acceptfd = 0;
    int max_fd = 0;
    int nbytes = 0;
    int i = 0;
    int j = 0;
    int ret = 0;
    //创建要监视的文件描述符集合
    struct pollfd fds[1024]; //每一个bit位监视一个文件描述符
    //清空集合 置-1的bit位不监视 只监视 1
    for (i = 0; i < 1024; i++) {
        fds[i].fd = -1;
    }
    //将sockfd添加到要监视的集合中
    fds[0].fd = sockfd;
    //需要监视 多个事件时  用 | 连接即可
    fds[0].events = POLLIN;
    //更新最大文件描述符
    max_fd = max_fd > sockfd ? max_fd : sockfd;

    while (1) {
        if (-1 == (ret = poll(fds, max_fd, 5000))) {
            ERRLOG("poll error");
        } else if (0 == ret) {
            printf("poll timeout ...\n");
            continue;
        } else {
            //说明有就绪的文件描述符了
            for (i = 0; i < max_fd && ret != 0; i++) {
                if (fds[i].revents & POLLIN != 0) { //按位 与 运算 1代表要监视
                    ret--;
                    if (fds[i].fd == sockfd) {
                        //说明有新的客户端连接了
                        if (-1 == (acceptfd = accept(fds[i].fd, NULL, NULL))) {
                            ERRLOG("accept error");
                        }
                        printf("客户端[%d]连接到服务器..\n", acceptfd);
                        //将acceptfd加入到集合中
                        //遍历数组 fds给acceptfd 找一个位置
                        for (j = 0; j < 1024; j++) {
                            if (fds[j].fd == -1) {
                                fds[j].fd = acceptfd;
                                fds[j].events = POLLIN;
                                break;
                            }
                        }
                        if (j == 1024) {
                            close(acceptfd);
                        }
                        //更新最大文件描述符
                        max_fd = max_fd > acceptfd ? max_fd : acceptfd;
                    } else {
                        //有客户端发来了数据
                        memset(buf, 0, sizeof(buf));
                        if (-1 == (nbytes = recv(fds[i].fd, buf, sizeof(buf), 0))) {
                            ERRLOG("recv error");
                        } else if (0 == nbytes) {
                            printf("客户端[%d]断开了连接..\n", fds[i].fd);
                            close(fds[i].fd);
                            fds[i].fd = -1;
                            continue;
                        }
                        if (!strcmp(buf, "quit")) {
                            printf("客户端[%d]退出了..\n", fds[i].fd);
                            close(fds[i].fd);
                            fds[i].fd = -1;
                            continue;
                        }
                        printf("客户端[%d]发来数据:[%s]\n", fds[i].fd, buf);
                        //组装应答消息
                        strcat(buf, "--hqyj");
                        //发送应答消息
                        if (-1 == send(fds[i].fd, buf, sizeof(buf), 0)) {
                            ERRLOG("send error");
                        }
                    }
                }
            }
        }
    }
    //关闭套接字
    close(sockfd);
    return 0;
}

4.客户端代码

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

#define ERRLOG(msg) do{\
        printf("%s %s %d:", __FILE__, __func__, __LINE__);\
        perror(msg);\
        exit(-1);\
}while(0)

#define N 128

int main(int argc, const char *argv[]){
    //入参合理性检查
    if(3 != argc){
        printf("Usage : %s <IP> <PORT>\n", argv[0]);
        return -1;
    }

    //1.创建套接字
    int sockfd = socket(AF_INET, SOCK_STREAM, 0);
    if(-1 == sockfd){
        ERRLOG("socket error");
    }

    //2.填充服务器网络信息结构体
    struct sockaddr_in serveraddr;
    memset(&serveraddr, 0, sizeof(serveraddr));
    serveraddr.sin_family = AF_INET;
    serveraddr.sin_port = htons(atoi(argv[2]));
    serveraddr.sin_addr.s_addr = inet_addr(argv[1]);

    socklen_t serveraddr_len = sizeof(serveraddr);

    char buff[128] = {0};
    int nbytes = 0;

    //3.尝试与服务器建立连接
    if(-1 == connect(sockfd, (struct sockaddr *)&serveraddr, serveraddr_len)){
        ERRLOG("connect error");
    }
    printf("与服务器建立连接成功..\n");
    while(1){
        memset(buff, 0, sizeof(buff));
        fgets(buff, N, stdin);
        buff[strlen(buff)-1] = '\0';//清理结尾的\n

        //发送数据
        if(-1 == send(sockfd, buff, sizeof(buff), 0)){
            ERRLOG("send error");
        }
        //接收服务器的应答信息
        if(-1 == (nbytes = recv(sockfd, buff, sizeof(buff), 0))){
            ERRLOG("recv error");
        }
        if(0 == nbytes){
            break;
        }
        //输出应答信息
        printf("应答为:[%s]\n", buff);
    }
    //关闭套接字
    close(sockfd);

    return 0;
}
  系统运维 最新文章
配置小型公司网络WLAN基本业务(AC通过三层
如何在交付运维过程中建立风险底线意识,提
快速传输大文件,怎么通过网络传大文件给对
从游戏服务端角度分析移动同步(状态同步)
MySQL使用MyCat实现分库分表
如何用DWDM射频光纤技术实现200公里外的站点
国内顺畅下载k8s.gcr.io的镜像
自动化测试appium
ctfshow ssrf
Linux操作系统学习之实用指令(Centos7/8均
上一篇文章      下一篇文章      查看所有文章
加:2022-09-13 11:56:34  更:2022-09-13 11:58:16 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/15 9:28:02-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码