Linux C++ 网络编程学习系列(3)——多路IO之poll实现
poll实现多路IO
- 源码地址:https://github.com/whuwzp/linuxc/tree/master/poll
- 源码说明:
- server.cpp: 监听127.1:6666,功能是将收到的小写转大写
- include/wrap.cpp: 封装的一些socket基本操作,加了基本的错误处理
1. 概要
一定先select实现,poll是select的改进版:,有以下不同
- 输入输出参数方面
- select: 需要allfds和readfds两个集合,因为readfds作为输入输出参数,每次select前都要用allfds赋值给它
- poll: pollfd结构体中,有events和revents两个,分别用来表示输入输出
- 监听集合的形式
- select: 用fd_set位图标记需要监听的fd
- poll: 用pollfd结构体中的fd成员标记,用结构体数组表示需要监听的集合
- 监听信号类型
- select: 用readfds, writefds集合分别表示
- poll: 用POLLIN,POLLOUT表示,
fds[0].events = POLLIN
- 判断是否有信号
- select: 用
FD_ISSET(fd,readfds)
- poll: 用
fds[0].revents & POLLIN
- select: 用
2. 核心代码
#include "include/wrap.h"
#include <arpa/inet.h>
#include <ctype.h>
#include <errno.h>
#include <netinet/in.h>
#include <poll.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;
struct pollfd fds[FD_SETSIZE];
int maxi = 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);
for (i = 0; i < FD_SETSIZE; ++i) {
fds[i].fd = -1;
fds[i].events = POLLIN;
}
fds[0].fd = fd_server;
maxi = 0;
while (true) {
printf("polling...\n");
nselect = poll(fds, (unsigned long)maxi+1, -1);
if (nselect == -1) {
if (errno == EINTR) {
continue;
} else {
perror_exit("poll failed");
}
}
printf("get %d select\n", nselect);
if (fds[0].revents & POLLIN) {
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 (fds[i].fd != -1) continue;
fds[i].fd = fd_client;
break;
}
printf("i: %d, FD_SETSIZE: %d\n", i, FD_SETSIZE);
if (i == FD_SETSIZE) perror_exit("too many clients");
if (i > maxi) maxi = i;
nselect--;
}
printf("going to find client, maxi: %d, nselect: %d\n", maxi, nselect);
for (i = 0; (i <= maxi) && (nselect > 0); ++i) {
if (!( fds[i].revents & POLLIN )) continue;
printf("find client %d\n", i + 1);
ret = workthread(fds[i].fd);
if (ret <= 0) {
Close(fds[i].fd);
fds[i].fd = -1;
}
nselect--;
}
sleep(3);
}
Close(fd_server);
for (i = 0; i < FD_SETSIZE; ++i) {
if (fds[i].fd == -1) continue;
Close(fds[i].fd);
}
}
3. 参考网址
Linux C++ 网络编程学习系列(3)——多路IO之poll实现的更多相关文章
- 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++ 网络编程学习系列(2)——多路IO之select实现
select实现多路IO 源码地址:https://github.com/whuwzp/linuxc/tree/master/select 源码说明: 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)-> ...
随机推荐
- MyBatis框架——动态SQL
MyBatis 作为⼀个“半⾃动化”的 ORM 框架,需要开发者⼿动定义 SQL 语句. 在业务需求⽐较复杂的情 况下,⼿动拼接 SQL 语句的⼯作量会⾮常⼤,为了适⽤于不同的业务需求,往往需要做很多 ...
- shiro拦截所有报 Uncaught SyntaxError: Unexpected token '<' 解决方法
改成 -> filterChainDefinitionMap.put("/css/**", "anon");filterChainDefinitionMa ...
- POJ1270 toposort+DFS+回溯
题目链接:http://poj.org/problem?id=1270 这道题其实就是求所有满足条件的topo序,我们考虑到给定的字符是确定的,也就是他们的长度都是一样的,所以为了得到所有的情况,我们 ...
- 洛谷1363 幻象迷宫dfs
题目网址:https://www.luogu.com.cn/problem/P1363 迷宫是无限多块地图拼接而成的,问是否可以在迷宫中走无限远.解决方案是dfs,走出初始地图之后的位置映射到原位置( ...
- 我们是怎么实现Grpc CodeFirst
前言: Grpc默认是ProtoFirst的,即先写 proto文件,再生成代码,需要人工维护proto,生成的代码也不友好,所以出现了Grpc CodeFirst,下面来说说我们是怎么实现Grpc ...
- pyplot 作图总结
折线图 下面是绘制折线图,设置图片的横轴纵轴标签,图片标题的API的用法. import matplotlib.pyplot as pyplot # init pyplot.figure() # ar ...
- PyTorch ImageNet 基于预训练六大常用图片分类模型的实战
微调 Torchvision 模型 在本教程中,我们将深入探讨如何对 torchvision 模型进行微调和特征提取,所有这些模型都已经预先在1000类的Imagenet数据集上训练完成.本教程将深入 ...
- Flume数据采集结合etcd作为配置中心在爬虫数据采集处理中的架构实践。
Apache Flume是一个分布式的.可靠的.可用的系统,用于有效地收集. 聚合和将大量日志数据从许多不同的源移动到一个集中的数据存储,但是其本身是以本地properties作为配置的,配置无法做到 ...
- Reface.AppStarter 框架初探
Reface.AppStarter 是一种基于 .NetFramework 的应用程序启动模式,使用该启动模式,你可以轻松的得到以下功能 : IOC / DI 自动注册与装配 简化配置 垂直模块化你的 ...
- Spring装配Bean的三种方式+导入和混合配置
目录 Spring IoC与bean 基于XML的显式装配 xml配置的基本结构 bean实例的三种创建方式 依赖注入的两种方式 构造器注入方式 setter方法注入方式 利用命名空间简化xml 基于 ...