Linux C++ 网络编程学习系列(2)——多路IO之select实现
select实现多路IO
- 源码地址:https://github.com/whuwzp/linuxc/tree/master/select
- 源码说明:
- server.cpp: 监听127.1:6666,功能是将收到的小写转大写
- include/wrap.cpp: 封装的一些socket基本操作,加了基本的错误处理
1. 概要
int select(int nfds, fd_set *restrict readfds,
fd_set *restrict writefds, fd_set *restrict errorfds,
struct timeval *restrict timeout);
void FD_CLR(int fd, fd_set *fdset);
int FD_ISSET(int fd, fd_set *fdset);
void FD_SET(int fd, fd_set *fdset);
void FD_ZERO(fd_set *fdset);
1.1 select函数
//原型
int select(int nfds, fd_set *restrict readfds,
fd_set *restrict writefds, fd_set *restrict errorfds,
struct timeval *restrict timeout);
//我的代码中的
nselect = select(fd_max + 1, &readfds, nullptr, nullptr, nullptr);
- fd_set类型数据: 用位图表示1024个fd(文件描述符)集合,sizeof看了该类型大小为128字节,也就是1024bit,应该是第n个bit表示fd为n的文件描述符,这也符合默认select最多可以监听1024个fd
- readfds,writefds,errorfds: 分别是读,写,错误的集合
- 以上的fds是输入输出参数,既是函数输入又是输出(以下以readfds为例)
- 输入: select依次去看readfds的1024bit,谁为1(或为0,这是猜想的),如果为1,假设第n个为1,那么select就会监听fd==n的文件描述符
- 输出: 如果监听的可读的话,那么就把这个bit置为1,假设第n个为1,则表示fd==n的文件描述符可读
- timeout: 就是超时返回, nullptr是永久等待
- nfds: 最大的文件描述符,应该是为了节约时间吧,这样就不用去轮询1024个fd,例如只有一个fd为5,那么只用轮询5前的几个,5之后就不用管了
注意:
- 由于fds是输入输出参数,所以每次select之后fds的内容都可能被修改,所以下次select前需要给fds重新赋值
- FD_SETSIZE默认就是1024
1.2 FD_xxx函数
void FD_CLR(int fd, fd_set *fdset);
int FD_ISSET(int fd, fd_set *fdset);
void FD_SET(int fd, fd_set *fdset);
void FD_ZERO(fd_set *fdset);
- FD_CLR: 清除fdset中的fd这个文件描述符,看看fd_set函数,这个实现应该很简单,就是直接第fd个bit置为0
- FD_ISSET: 判断fd是不是在fdset中,这个用于判断select之后,fd是否有信号到来,实现也很简单,就是看看第fd个bit是不是1
- FD_SET: 这个是把fd加入到fdset中,一般在select前,设置需要监听的fds
- FD_ZERO: 清零
2. 核心代码
#include "include/wrap.h"
#include <arpa/inet.h>
#include <ctype.h>
#include <errno.h>
#include <netinet/in.h>
#include <pthread.h>
#include <stdio.h>
#include <stdlib.h>
#include <stdnoreturn.h>
#include <string.h>
#include <sys/select.h>
#include <sys/socket.h>
#include <sys/types.h> /* See NOTES */
#include <unistd.h>
#include <wait.h>
#define LOCALIP "127.0.0.1"
#define PORT 6666
void handler(char *in, char *out) {
for (int i = 0; i < (int)strlen(out) + 1; ++i) {
out[i] = toupper(in[i]);
}
}
int workthread(const int &fd_client) {
char recvbuf[2048] = {0};
char sendbuf[2048] = {0};
int ret = 0;
ret = (int)Read(fd_client, recvbuf, 2048);
if (ret <= 0) {
printf("ret==0\n");
return ret;
}
handler(recvbuf, sendbuf);
ret = (int)Write(fd_client, sendbuf, strlen(sendbuf) + 1);
return ret;
}
void startsock(int &fd, struct sockaddr_in &addr, const char *ip,
const int port) {
fd = Socket(AF_INET, SOCK_STREAM, 0);
memset(&addr, 0, sizeof(addr));
addr.sin_family = AF_INET;
addr.sin_addr.s_addr = inet_addr(ip);
addr.sin_port = htons(port);
}
int main() {
int fd_server = 0;
int fd_client = 0;
int ret = 0;
struct sockaddr_in sock_client;
struct sockaddr_in sock_server;
socklen_t client_len = (socklen_t)sizeof(sock_client);
int opt = 0;
fd_set readfds;//作为select的参数
fd_set allfds;//保存fds,因为readfds会被修改, 见 1.1 select 注意
int fd_clients[FD_SETSIZE]; //用于保存客户端连接的fd
memset(fd_clients, -1, sizeof(fd_clients)); //用-1表明是没有用过的fd,方便后面找空位
int maxi = 0;
int fd_max = 0;
int nselect = 0;
int i = 0;
startsock(fd_server, sock_server, LOCALIP, PORT);
opt = 1;
Setsockopt(fd_server, SOL_SOCKET, SO_REUSEADDR, &opt,
(socklen_t)sizeof(opt));
Bind(fd_server, (struct sockaddr *)&sock_server, sizeof(sock_server));
Listen(fd_server, 5);
FD_ZERO(&allfds);
FD_ZERO(&readfds);
FD_SET(fd_server, &allfds);//把listenfd加入待监听的
fd_max = fd_server;
while (true) {
readfds = allfds; //见 1.1 select 注意
printf("selecting...\n");
nselect = select(fd_max + 1, &readfds, nullptr, nullptr, nullptr);
if (nselect == -1) {
if (errno == EINTR) {
continue;
} else {
perror_exit("select failed");
}
}
printf("get %d select\n", nselect);
if (FD_ISSET(fd_server, &readfds)) { //看看listenfd有没有信号,就是有没有新连接
fd_client =
Accept(fd_server, (struct sockaddr *)&sock_client, &client_len);
printf("accept: %s: %d\n", inet_ntoa(sock_client.sin_addr),
ntohs(sock_client.sin_port));
for (i = 0; i < FD_SETSIZE; ++i) {
if (fd_clients[i] != -1) continue; //找数组中没有用过的空位存fd_client
fd_clients[i] = fd_client;
break;
}
printf("i: %d, FD_SETSIZE: %d\n", i, FD_SETSIZE);
if (i == FD_SETSIZE) perror_exit("too many clients");//超过1024
if (i > maxi) maxi = i;
if (fd_client > fd_max) fd_max = fd_client;//找最大的fd
FD_SET(fd_client, &allfds);//加入监听集合
nselect--;
}
printf("going to find client, maxi: %d, nselect: %d\n", maxi, nselect);
for (i = 0; (i <= maxi) && (nselect > 0); ++i) {//maxi为了节约时间,不用轮询1024次
if (FD_ISSET(fd_clients[i], &readfds) == 0) continue;//轮询哪个clientfd有信号
printf("find client %d\n", i + 1);
ret = workthread(fd_clients[i]);//处理
if (ret <= 0) {//断开
Close(fd_clients[i]);
FD_CLR(fd_clients[i], &allfds);
fd_clients[i] = -1;
}
nselect--;
}
sleep(3);
}
Close(fd_server);
for (i = 0; i < FD_SETSIZE; ++i) {
if (fd_clients[i] == -1) continue;
Close(fd_clients[i]);
}
}
注意: 代码中maxi和fd_max的操作看上去很繁琐,其实都是为了节约时间,例如
- maxi是标记当前fd_clients中用到的最大索引值,这样我们找哪个fd有信号时,就不用轮询fd_clients中1024个fd了;
- fd_max是最大的fd,这样nfds就可以方便设置,select就不用监听1024个
- 如果觉得繁琐,其实可以全部改成1024,就是浪费
3. 参考网址
Linux C++ 网络编程学习系列(2)——多路IO之select实现的更多相关文章
- Linux C++ 网络编程学习系列(1)——端口复用实现
Linux C++ 网络编程学习系列(1)--端口复用实现 源码地址:https://github.com/whuwzp/linuxc/tree/master/portreuse 源码说明: serv ...
- Linux C++ 网络编程学习系列(6)——多路IO之epoll高级用法
poll实现多路IO 源码地址:https://github.com/whuwzp/linuxc/tree/master/epoll_libevent 源码说明: server.cpp: 监听127. ...
- Linux C++ 网络编程学习系列(5)——多路IO之epoll边沿触发
多路IO之epoll边沿触发+非阻塞 源码地址:https://github.com/whuwzp/linuxc/tree/master/epoll_ET_LT_NOBLOCK_example 源码说 ...
- Linux C++ 网络编程学习系列(4)——多路IO之epoll基础
epoll实现多路IO 源码地址:https://github.com/whuwzp/linuxc/tree/master/epoll 源码说明: server.cpp: 监听127.1:6666,功 ...
- Linux C++ 网络编程学习系列(3)——多路IO之poll实现
poll实现多路IO 源码地址:https://github.com/whuwzp/linuxc/tree/master/poll 源码说明: server.cpp: 监听127.1:6666,功能是 ...
- Linux C++ 网络编程学习系列(7)——mbedtls编译使用
mbedtls编译使用 环境: Ubuntu18.04 编译器:gcc或clang 编译选项: 静态编译使用 1. mbedtls源码 下载地址: https://github.com/ARMmbed ...
- Linux C网络编程学习笔记
Linux C网络编程总结报告 一.Linux C 网络编程知识介绍: 网络程序和普通的程序有一个最大的区别是网络程序是由两个部分组成的--客户端和服务器端. 客户端:(client) 在网络程序中, ...
- linux下网络编程学习——入门实例ZZ
http://www.cppblog.com/cuijixin/archive/2008/03/14/44480.html 是不是还对用c怎么实现网络编程感到神秘莫测阿,我们这里就要撕开它神秘的面纱, ...
- Linux下网络编程学习杂记
1.TCP/IP协议的体系结构包含四层:应用层(负责应用程序的网络服务,通过端口号识别各个不同的进程)->传输层(传输控制层协议TCP.用户数据报协议UDP.互联网控制消息协议ICMP)-> ...
随机推荐
- 基于Ubuntu的ORB-SLAM2项目环境搭建过程
目录 关于ORB-SLAM2 环境搭建 已有环境 创建环境 新建项目目录 安装Pangolin 安装OpenCV 3.2 安装Eigen DBoW2 and g2o (Included in Thir ...
- redis 持久化RDB、AOF
1.redis持久化简介 Redis是一种高级key-value数据库.它跟memcached类似,不过数据可以持久化,而且支持的数据类型很丰富.有字符串,链表,集合和有序集合.支持在服务器端计算集合 ...
- C# 微信 生活助手 空气质量 天气预报等 效果展示 数据抓取 (二)
此文主要是 中国天气网和中国环境监测总站的数据抓取 打算开放全部数据抓取源代码 已在服务器上 稳定运行半个月 webapi http://api.xuzhiheng.cn/ 常量 /// <su ...
- css3笔记系列-3.css中的各种选择器详解,不看后悔系列
点击上方蓝色字体,关注我 最详细的css3选择器解析 选择器是什么? 比较官方的解释:在 CSS 中,选择器是一种模式,用于选择需要添加样式的元素. 最常见的 CSS 选择器是元素选择器.换句话说 ...
- 最简易 Pair of Topics解决方法
这个题花费了我两天的时间来解决,最终找到了两个比较简单的方法 首先这个题不难看出是寻找a[i]+a[j]<0的情况,我第一开始直接用两个for循环遍历通过不了,应该是复杂度太大了 第一个方法 # ...
- 直径问题 Diameter Problems
2019-11-03 21:37:59 一.Diameter of Binary Tree 问题描述: 问题求解: 解法一.第一反应是树上动归,每个节点保存一下左右的最大深度,最后以每个节点作为中枢计 ...
- [STL] Codeforces 69E Subsegments
Subsegments time limit per test 1 second memory limit per test 256 megabytes input standard input ou ...
- 浏览器与DNS解析过程
浏览器解析 1.地址栏输入地址后,浏览器检查自身DNS缓存 地址栏输入chrome://net-internals/#dns 查看. 2.浏览器缓存中未找到,那么Chrome会搜索操作系统自身的DNS ...
- Pandas和Numpy的一些金融相关的操作(一)
Pandas和Numpy的一些金融相关的操作 给定一个净值序列,求出最大回撤 # arr是一个净值的np.ndarray i = np.argmax( (np.maximum.acumulate(ar ...
- Colab笔记本能用英伟达Tesla T4了,谷歌的羊毛薅到酸爽
谷歌出品的Colab笔记本,机器学习界薅羊毛神器,如今又有了新福利: 连英伟达最新一代机器学习GPU:Tesla T4都能免费蹭,穷苦羊毛党也顿时高端了起来. 英伟达的Tesla T4,是去年秋天才发 ...