【网络编程】poll
主旨思想
- 用一个结构体记录文件描述符集合,并记录用户态状态和内核态状态
函数说明
- 概览
#include <poll.h>
struct pollfd { int fd; /* 委托内核检测的文件描述符 */ short events; /* 委托内核检测文件描述符的什么事件 */ short revents; /* 文件描述符实际发生的事件 */
};int poll(struct pollfd *fds, nfds_t nfds, int timeout);
- int poll(struct pollfd *fds, nfds_t nfds, int timeout);
通过man poll查看帮助
参数
fds:是一个struct pollfd 结构体数组,这是一个需要检测的文件描述符的集合
nfds:这个是第一个参数数组中最后一个有效元素的下标 + 1
timeout:阻塞时长
0:不阻塞
-1:阻塞,当检测到需要检测的文件描述符有变化,解除阻塞
>0:具体的阻塞时长(ms)
返回值
-1:失败
>0(n):检测的集合中有n个文件描述符发生了变化
events及revents取值,如果有多个事件需要检测,用|即可,如同时检测读和写:POLLIN|POLLOUT
代码实现
注意事项
nfds
表示的监听文件描述符的下标,所以在遍历时,需要使用fds[i].fd
取得相应的文件描述符- 如何优雅的更新nfds?代码中使用连接的文件描述符作为替代更新
服务器端:
#include <stdio.h>
#include <arpa/inet.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <poll.h>#define SERVERIP "127.0.0.1"
#define PORT 6789int main()
{// 1. 创建socket(用于监听的套接字)int listenfd = socket(AF_INET, SOCK_STREAM, 0);if (listenfd == -1) {perror("socket");exit(-1);}// 2. 绑定struct sockaddr_in server_addr;server_addr.sin_family = PF_INET;// 点分十进制转换为网络字节序inet_pton(AF_INET, SERVERIP, &server_addr.sin_addr.s_addr);// 服务端也可以绑定0.0.0.0即任意地址// server_addr.sin_addr.s_addr = INADDR_ANY;server_addr.sin_port = htons(PORT);int ret = bind(listenfd, (struct sockaddr*)&server_addr, sizeof(server_addr));if (ret == -1) {perror("bind");exit(-1);}// 3. 监听ret = listen(listenfd, 8);if (ret == -1) {perror("listen");exit(-1);}struct pollfd fds[1024];// 初始化for (int i = 0; i < 1024; i++) {fds[i].fd = -1;fds[i].events = POLLIN;}// 将监听文件描述符加入fds[0].fd = listenfd;int nfds = 0;// 不断循环等待客户端连接while (1) {// 使用poll,设置为永久阻塞,有文件描述符变化才返回int num = poll(fds, nfds + 1, -1);if (num == -1) {perror("poll");exit(-1);} else if (num == 0) {// 当前无文件描述符有变化,执行下一次遍历// 在本次设置中无效(因为select被设置为永久阻塞)continue;} else {// 首先判断监听文件描述符是否发生改变(即是否有客户端连接)if (fds[0].revents & POLLIN) {// 4. 接收客户端连接struct sockaddr_in client_addr;socklen_t client_addr_len = sizeof(client_addr);int connfd = accept(listenfd, (struct sockaddr*)&client_addr, &client_addr_len);if (connfd == -1) {perror("accept");exit(-1);}// 输出客户端信息,IP组成至少16个字符(包含结束符)char client_ip[16] = {0};inet_ntop(AF_INET, &client_addr.sin_addr.s_addr, client_ip, sizeof(client_ip));unsigned short client_port = ntohs(client_addr.sin_port);printf("ip:%s, port:%d\n", client_ip, client_port);// 遍历集合, 将新的需要监听的文件描述符加入集合for (int i = 1; i < 1024; i++) {if (fds[i].fd == -1) {fds[i].fd = connfd;fds[i].events = POLLIN;break;}}// 更新最大的监听文件描述符集合下标// 存在问题:使用文件描述符替代最大对应下标nfds = nfds > connfd ? nfds : connfd;}// 遍历集合判断是否有变动,如果有变动,那么通信char recv_buf[1024] = {0};for (int i = 1; i <= nfds; i++) {if (fds[i].fd != -1 && fds[i].revents & POLLIN) {ret = read(fds[i].fd, recv_buf, sizeof(recv_buf));if (ret == -1) {perror("read");exit(-1);} else if (ret > 0) {printf("recv server data : %s\n", recv_buf);write(fds[i].fd, recv_buf, strlen(recv_buf));} else {// 表示客户端断开连接printf("client closed...\n");close(fds[i].fd);fds[i].fd = -1;break;}}}}}close(listenfd);return 0;
}
客户端:
#include <stdio.h>
#include <arpa/inet.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>#define SERVERIP "127.0.0.1"
#define PORT 6789int main()
{// 1. 创建socket(用于通信的套接字)int connfd = socket(AF_INET, SOCK_STREAM, 0);if (connfd == -1) {perror("socket");exit(-1);}// 2. 连接服务器端struct sockaddr_in server_addr;server_addr.sin_family = PF_INET;inet_pton(AF_INET, SERVERIP, &server_addr.sin_addr.s_addr);server_addr.sin_port = htons(PORT);int ret = connect(connfd, (struct sockaddr*)&server_addr, sizeof(server_addr));if (ret == -1) {perror("connect");exit(-1);}// 3. 通信char recv_buf[1024] = {0};while (1) {// 发送数据char *send_buf = "client message";write(connfd, send_buf, strlen(send_buf));// 接收数据ret = read(connfd, recv_buf, sizeof(recv_buf));if (ret == -1) {perror("read");exit(-1);} else if (ret > 0) {printf("recv server data : %s\n", recv_buf);} else {// 表示客户端断开连接printf("client closed...\n");}// 休眠的目的是为了更好的观察,放在此处可以解决read: Connection reset by peer问题sleep(1);}// 关闭连接close(connfd);return 0;
}
存在问题(缺点)
- 缺点同
select
第一点和第二点(如下),即解决了第三点和第四点 - 每次调用select,都需要把fd集合从用户态拷贝到内核态,这个开销在fd很多时会很大
- 同时每次调用select都需要在内核遍历传递进来的所有fd,这个开销在fd很多时也很大
相关文章:

【网络编程】poll
主旨思想 用一个结构体记录文件描述符集合,并记录用户态状态和内核态状态 函数说明 概览 #include <poll.h> struct pollfd { int fd; /* 委托内核检测的文件描述符 */ short events; /* 委托内核检测文件描述符的什么事件 */ short revents; /* 文件描述…...

配置VS Code 使其支持vue项目断点调试
起因 每个应用,不论大小,都需要理解程序是如何运行失败的。当我们写的程序没有按照自己写的逻辑走的时候,我们就会逐步一一排查问题。在平常开发过程中我们可能会借助 console.log 来排查,但是现在我们可以借助 VS Code 断点来调试项目。 前…...

第一百零一回 如何在组件树之间共享数据
文章目录 概念介绍使用方法示例代码 我们在上一章回中介绍了"如何实现文件存储"相关的内容,本章回中将介绍 如何实现组件之间共享数据。闲话休提,让我们一起Talk Flutter吧。 概念介绍 数据共享是程序中常用的功能,本章回介绍如何…...

Golang进阶学习
Golang进阶学习 视频地址:https://www.bilibili.com/video/BV1Pg41187AS?p35 1、包 1.1、包的引入 使用包的原因: 我们不可能把所有函数放在同一个源文件中,可以分门别类的放在不同的文件中 解决同名问题,同一个文件中不可以…...

【Linux】常用的基本指令
👦个人主页:Weraphael ✍🏻作者简介:目前正在学习c和算法 ✈️专栏:Linux 🐋 希望大家多多支持,咱一起进步!😁 如果文章有啥瑕疵,希望大佬指点一二 如果文章对…...

栈溢出几种情况及解决方案
一、局部数组过大。当函数内部的数组过大时,有可能导致堆栈溢出。 二、递归调用层次太多。递归函数在运行时会执行压栈操作,当压栈次数太多时,也会导致堆栈溢出。 三、指针或数组越界。这种情况最常见,例如进行字符串拷贝&#…...

go 内存分配
关注 go 语言内存分配策略,主要是想了解 go 的性能。申请不同大小的内存,性能开销是有差别的,申请内存越大,耗时也越久,性能也越差。 内存分配 参考 Go1.17.13 版本源码,从内存分配大小上区分了 tiny、sm…...

Maven pom.xml文件中build,plugin标签的具体使用
<build> 标签 <build> 标签是 pom.xml 文件中一个重要的标签,用于配置 Maven 项目的构建过程。在 <build> 标签下,可以配置构建相关的设置,包括源代码目录、输出目录、插件配置等。 以下是 <build> 标签的详细使用方…...

批量插入数据、MVC三层分离
八、批量插入数据 1、使用Statement() 2、使用PreparedStatement() 3、使用批量操作API 4、优化 九、MVC三层分离...

【IMX6ULL驱动开发学习】21.Linux驱动之PWM子系统(以SG90舵机为例)
1.设备树部分 首先在 imx6ull.dtsi 文件中已经帮我们定义好了一些pwm的设备树节点,这里以pwm2为例 pwm2: pwm02084000 {compatible "fsl,imx6ul-pwm", "fsl,imx27-pwm";reg <0x02084000 0x4000>;interrupts <GIC_SPI 84 IRQ_TYP…...

el-cascader级联选择器加载远程数据、默认开始加载固定条、可以根据搜索加载远程数据。
加载用户列表分页请求、默认请求20条数据。想添加远程搜索用户功能。原有的方法filter-method不能监听到输入清空数据的时候。这样搜索完无法返回默认的20条数据。直接监听级联选择的v-model绑定的值是无法检测到用户自己输入的。 解决思路: el-cascader 没有提供…...

大数据技术之Clickhouse---入门篇---SQL操作、副本
星光下的赶路人star的个人主页 积一勺以成江河,累微尘以崇峻极 文章目录 1、SQL操作1.1 Insert1.2 Update 和 Delete1.3 查询操作1.4 alter操作1.5 导出数据 2、副本2.1 副本写入流程2.2 配置步骤 1、SQL操作 基本上来说传统关系型数据库(以 MySQL 为例…...

【Rust 基础篇】Rust Sized Trait:理解Sized Trait与动态大小类型
导言 Rust是一门以安全性和性能著称的系统级编程语言。在Rust中,类型大小的确定在编译期是非常重要的。然而,有些类型的大小在编译期是无法确定的,这就涉及到了Rust中的动态大小类型(DST)。为了保证在编译期可以确定类…...

前端框架学习-Vue(三)
目录 初识VueVue模板语法数据绑定el和data的两种写法事件的基本使用$emit在子组件中定义方法,执行父组件的方法 Vue中的事件修饰符:键盘事件计算属性监视属性条件渲染列表渲染表单数据收集过滤器 笔记内容来自:尚硅谷Vue2.0Vue3.0全套教程丨v…...

HTML <rt> 标签
实例 一个 ruby 注释: <ruby> 漢 <rt> ㄏㄢˋ </rt> </ruby>浏览器支持 元素ChromeIEFirefoxSafariOpera<rt>5.05.538.05.015.0 Internet Explorer 9, Firefox, Opera, Chrome 以及 Safari 支持 <rt> 标签。 注释…...

VMware Linux Centos 配置网络并设置为静态ip
在root用户下进行以下操作 1. 查看子网ip和网关 (1)进入虚拟网络编辑器 (2)进入NAT设置 (3)记录子网IP和子网掩码 2. 修改网络配置文件 (1)cd到网络配置文件路径下 [rootlo…...

【Leetcode 30天Pandas挑战】学习记录
这个系列难度比较低,一题写一篇其实没必要,就全部放到一篇吧 题目列表: 595. Big Countries1757. Recyclable and Low Fat Products 595. Big Countries 原题链接:595. Big Countries Table: World ---------------------- | C…...

微信小程序使用 canvas 2d 实现签字板组件
本文是在微信小程序中使用 canvas 2d 来实现签字板功能; 效果图: 代码: 1、wxml <view><canvas id"canvas"type"2d"bindtouchstart"start"bindtouchmove"move"bindtouchend"end&qu…...

区块链赋能新时代司法体系,中移链打造可信存证服务
近期,某百万级粉丝网红的法律维权之路引发社会关注。其在面对网络造谣行为时积极搜集证据,使用区块链技术将相关信息上链保全,然后将造谣者全部起诉,一系列操作被广大网友喻为是教科书式网络维权。 科技在发展,时代在…...

ELK报错no handler found for uri and method [PUT] 原因
执行后提示no handler found for uri and method post,最新版8.2的问题? 原因: index.mapping.single_type: true在索引上 设置将启用按索引的单一类型行为,该行为将在6.0后强制执行。 原 {type} 要改为 _doc,格式如…...

Sublime操作技巧笔记
同时选中2个文件:自动切换成左右2个界面 格式化代码ctrlshifth: 使用快捷键ctrl shift p调出控制台,输入install package,然后输入html-css-js prettify,进行下载。具体的快捷键在preference > package setting &g…...

JVM | 基于类加载的一次完全实践
引言 我在上篇文章:JVM | 类加载是怎么工作的 中为你介绍了Java的类加载器及其工作原理。我们简单回顾下:我用一个易于理解的类比带你逐步理解了类加载的流程和主要角色:引导类加载器,扩展类加载器和应用类加载器。并带你深入了解…...

Termux实现电脑端远程操作【开启SSH的完整教程】
文章目录 前言一、安装软件1、安装2、启动服务3、特别说明4、添加key二、电脑端连接1、查看ip2、电脑端连接总结前言 上篇文章【安卓手机变身Linux服务器】讲了如何将你的上古安卓手机变废为宝,这节着重为大家解决一个痛点:“手机上操作实在是不方便”。 一、安装软件 1、安…...

java(Collection类)
文章目录 Collection接口继承树Collection接口及方法判断删除其它 Iterator(迭代器)接口迭代器的执行原理 foreach循环Collection子接口1:ListList接口特点List接口方法List接口主要实现类:ArrayListList的实现类之二:LinkedListList的实现类…...

VS2019编译安装OpenMesh8.0
文章目录 一、简介二、相关准备三、编译安装四、举个栗子参考资料一、简介 多边形网格一直以来就是交互式3D图形应用程序中最合适的几何表示,它们足够灵活,可以近似任意形状,并且可以通过当前的图形硬件有效地处理,即使在今天的低成本电脑上也是如此。OpenMesh便是其中一种…...

Python爬虫遇到URL错误解决办法大全
在进行Python爬虫任务时,遇到URL错误是常见的问题之一。一个错误的URL链接可能导致爬虫无法访问所需的网页或资源。为了帮助您解决这个问题,本文将提供一些实用的解决方法,并给出相关代码示例,希望对您的爬虫任务有所帮助。 一、…...

基于Vue+ElementUI+Echarts+G2Plot的大屏设计器,代码完全开源
简介 🔥DataRoom是一款基于SpringBoot、MyBatisPlus、ElementUI、G2Plot、Echarts等技术栈的大屏设计器,具备大屏设计、预览、资源管理、组件管理等能力,支持JSON、MySQL、Oracle、PostgreSQL、HTTP、JavaScript、Groovy等数据集接入&#x…...

Linux - PostgreSQL 适用于9.x 以上的 tar.gz 源码安装与理解 - 报错集锦
这里写目录标题 序言主要内容bash 配置文件个人理解关于初始化 PostgreSQL 数据库的理解 启动方法检查服务器是否在PostgreSQL中运行关闭 postgresql 数据库方法参考链接 序言 PostgreSQL 9.x 以下版本笔者没用过,具体操作看参考链接,笔者就不记录重复操…...

Django使用用户列表的展示和添加
接着上一篇:https://blog.csdn.net/javascript_good/article/details/132027702 来实现用户表的查询和添加 1、创建数据库表 在models.py 中,增加UserInfo类,包括字段姓名、密码、年龄、账号余额、入职时间、所属部门、性别 verbose_name 就…...

kubernetes错误汇总
title: “kubernetes错误汇总” categories: - “技术” tags: - “Kubernetes” - “错误汇总” toc: false original: true draft: false 1、增加 master etcd 报错 1.1、错误描述 由于创建的k8s集群,其中有一个master节点初始化失败,先删除了这个节…...