关于socket编程中FD_XXX以及select函数的理解

news2024/12/28 18:15:55

文章目录

  • 01 | 宏接口定义
  • 02 | 使用方法
  • 03 | 服务端代码示例

在这里插入图片描述

学习socket编程的时候看到很多FD开头的宏定义和函数,这里记录一下这些宏定义和函数的含义及处理流程

01 | 宏接口定义

  1. fd_set

    fd_set 是一种表示文件描述符的集合类型,在socket编程中,这种类型有三种不同的集合(可读、可写、错误)

  2. FD_ZERO() | FD_CLR()

    • FD_ZERO(fd_set* fdset):把当前描述符集合 fd_set 中所有位的数字都置为0

    • FD_CLR(int fd, fd_set* fdset):清楚所绑定的联系,将 fd 从 fd_set 集合中清除,类似于链表的节点删除(后续节点会填补被删除节点)

  3. FD_SET()

    FD_SET(int fd, fd_set* fdset):实现了句柄和描述符集合的联系,可以把已打开的 fd(句柄)加入到描述符集合 fd_set 中

  4. select()

    用于查询所有描述符(句柄)所处的状态,获取当中可读可写可用的描述符个数

    int select(int maxfdpl, fd_set *restrict readfds,
    		fd_set *restrict writefds,fd_set *restrict exceptfds,
    		struct timeval *restrict typfr);
    // 返回值:准备就绪的描述符数目;若超时返回0,出错则返回-1
    

    参数说明:

    • maxfdpl:最大描述符编号 + 1,即选择需要关注的描述符个数,一般为环境中可能打开的描述符个数加一

    • readfds | writefds | exceptfds:这三个指向描述符的指针,分别代表了所要关心的 可读描述符集 | 可写描述符集 | 出错描述符集

    • typfr:超时时间

  5. FD_ISSET()

    FD_ISSET(int fd, fd_set* fdset):判断加入的 fd 是否还存在于描述符集合 fd_set 中,即判断打开的 fd 是否可用。不存在或不可用则返回0

02 | 使用方法

总体使用流程就是:通过调用 FD_SERO() 将一个 fd_set 变量的所有位设置为0(如果要开启描述符集合中的一位,可以调用 FD_SET();如果要清楚刚才设置的位则可以调用 FD_CLR());再通过调用 select() 对描述符集合进行筛选,得到当前可用描述符个数;最后可以调用 FD_ISSET() 测试当前描述符集合中的一个指定位是否已打开可用

简要流程如下

  1. 初始化一个 fd_set 类型的描述符集合

    fd_set server_fd;
    FD_ZERO(&server_fd);
    
  2. 把 socket调用打开的 fd 放入到已初始化的描述符集合中

    FD_SET(socket_fd, &server_fd);
    
  3. 通过调用 select() 查询这个描述符集合中的所有 fd 状态,并清除描述符集合中 不可读 | 不可写 的 fd,或者说选出 可读 | 可写 的 fd

    int max_fd = -1;
    if (max_fd < socket_fd)
    {
    	max_fd = socket_fd;
    }
    if (select(max_fd + 1, &server_fd, NULL, NULL, &timeout) < 0)
    {
    	return;
    }
    
  4. 判断当前打开的 fd 是否还存在于描述符集合中

    if (FD_ISSET(socketfd, &server_fd))
    {
    // To do
    }
    
  5. 上述步骤完成后,还可以再进行错误处理判断,防止后续错误

    int iRet = 0; 
    int len = sizeof(iRet); 
    if (getsockopt(m_sock, SOL_SOCKET, SO_ERROR, (char*)&iRet, &len) == SOCKET_ERROR) 
    { 
    	return; 
    } 
    

03 | 服务端代码示例

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

int main()
{
	int lfd = socket(AF_INET, SOCK_STREAM, 0);
	if(lfd < 0)
	{
		perror("socket error");
		return -1;
	}

	struct sockaddr_in serv;
	bzero(&serv, sizeof(serv));
	serv.sin_family = AF_INET;
	serv.sin_port = htons(8888);
	serv.sin_addr.s_addr = htonl(INADDR_ANY);
	int ret = bind(lfd, (struct sockaddr*)&serv, sizeof(serv));
	if(ret < 0)
	{
		perror("bind error");
		return -1;
	}

	listen(lfd, 128);

	fd_set readset;
	fd_set tmpset;
	FD_ZERO(&readset);
	FD_SET(lfd, &readset);

	int nready;
	int maxfd = lfd;
	int cfd;
	int n;
	char buf[1024];
	while(1)
	{
		tmpset = readset;
		nready = select(maxfd+1, &tmpset, NULL, NULL, NULL);
		if(nready <= 0)
		{
			continue;
		}

		//有客户端连接请求到来
		if(FD_ISSET(lfd, &tmpset))
		{
			cfd = accept(lfd, NULL, NULL);
			FD_SET(cfd, &readset);
			maxfd = cfd > maxfd? cfd : maxfd;
			//说明只有监听描述符被置为1,后面可跳过
			if(--nready == 0)
			{
				continue;
			}
		}

		//有数据发来
		//先有的lfd,后有的cfd
		for(int i=lfd+1; i<=maxfd;i++)
		{
			if(FD_ISSET(i, &tmpset))
			{
				memset(buf, 0, sizeof(buf));
				n = read(cfd, buf, sizeof(buf));
				if(n <= 0)
				{
					close(i);
					FD_CLR(i, &readset);
					printf("read error or client close\n");
					continue;
				}
				printf("n == [%d], buf == [%s]\n", n, buf);

				for(int j=0; j<n; j++)
				{
					buf[j] = toupper(buf[j]);
				}

				write(i, buf, n);
			}
		}
	}

	close(lfd);
	return 0;
}
各位大佬点点关注,点赞,收藏,有空的时候再回来看看,谢谢

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/674313.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

计算机网络——自顶向下方法(第三章学习记录)

本章学习运输层 运输层位于应用层和网络层之间&#xff0c;是分层的网络体系的重要部分&#xff0c;该层为运行在不同主机上的应用进程提供直接的通信服务起着至关重要的作用。 运输层协议为运行在不同主机上的应用进程之间提供了逻辑通信(logic communication)功能。从应用程…

CSS3-补充-伪元素

伪元素 作用&#xff1a;在网页中创建非主体内容&#xff0c;开发中常用CSS创建标签&#xff0c;比如装饰性的不重要的小图 区别&#xff1a; 1 元素&#xff1a;HTML 设置的标签 2 伪元素&#xff1a;由 CSS 模拟出的标签效果 …

EMC学习笔记(七)阻抗控制(一)

阻抗控制&#xff08;一&#xff09; 1.特征阻抗的物理意义1.1 输入阻抗1.2 特征阻抗1.3 偶模阻抗、奇模阻抗、差分阻抗 2.生产工艺对阻抗控制的影响 1.特征阻抗的物理意义 1.1 输入阻抗 在集总电路中&#xff0c;输入阻抗是经常使用的一个术语 &#xff0c;它的物理意义是: …

FreeRTOS实时操作系统(六)列表与列表项

系列文章目录 文章目录 系列文章目录简要概念列表列表项迷你列表项 相关API函数初始化列表列表项初始化列表项插入&#xff08;升序&#xff09;末尾列表项插入列表项删除 实战实验 简要概念 列表是 FreeRTOS 中的一个数据结构&#xff0c;概念上和链表有点类似&#xff0c;列…

ubuntu环境下测试硬盘读写速度

在Ubuntu下&#xff0c;可以使用hdparm、dd和fio等工具来测试硬盘的读写速度。 开始之前&#xff0c;先使用sudo fdisk -l命令来列出系统中所有的硬盘和分区&#xff1a; 1.使用hdparm测试硬盘读取速度&#xff1a; 安装hdparm&#xff1a; sudo apt-get install hdparm 通…

C++17中utf-8 character literal的使用

一个形如42的值被称作字面值常量(literal),这样的值一望而知。每个字面值常量都对应一种数据类型&#xff0c;字面值常量的形式和值决定了它的数据类型。 由单引号括起来的一个字符称为char型字面值&#xff0c;双引号括起来的零个或多个字符则构成字符串型字面值。 字符串字面…

9.QT 三目运算符

上面引出两个新的概念&#xff1a; 左值&#xff1a;能被赋值的就是左值。 右值&#xff1a;不能被赋值的就是右值。

基於ranger,kerberos,hadoop ha 配置hvie多用戶

基於ranger&#xff0c;kerberos&#xff0c;hadoop ha 配置hvie多用戶 hive多用戶權限管理一、hive的管理員用戶二、hive配置普通用戶1.添加用戶2.配置kerberos2.1 创建主体2.2 生成keytab文件2.3 修改keytab文件所有者(可做可不做) 3. 配置windows hive多用戶權限管理 一、h…

English Learning - L3 综合练习 8 TED-Living Beyond the Limits 2023.06.21 周三

English Learning - L3 综合练习 8 TED-Living Beyond the Limits 2023.06.21 周三 句 1句 2扩展 句 3句 4句 5句 6句 7扩展 random 句 8扩展 句 9句 10句 11句 12句 13句 14句 15句 16句 17句 18句 19句 20句 21句 22句 23 句 1 Four months later I was back up on a snowbo…

SVN使用步骤

1.基本操作 2.提交之间看一下变更内容 3.显示日志 是查看所有提交的记录4.撤销和恢复操作 撤销本地修改 或者点击提交的时候 还原 把修改的撤销掉 第二种情况,内容已经提交上去了点击提交日志 进行操作 只是撤销了本地 接着还需要继续提交到服务端 第三种情况 我们需要恢…

Linux系统之安装showdoc文档工具

Linux系统之部署showdoc文档工具 一、showdoc介绍1.1 showdoc简介1.2 showdoc功能 二、本地环境介绍2.1 本地环境规划2.2 本次实践介绍 三、检查本地环境3.1 检查本地操作系统版本3.2 检查系统内核版本3.3 检查本地yum仓库状态 四、安装httpd服务4.1 安装httpd4.2 启动httpd服务…

用Visual C++写出你第一个Windows程序

我是荔园微风&#xff0c;作为一名在IT界整整25年的老兵&#xff0c;今天来看看如何用Visual C写出你第一个Windows程序。 与其看很多Windows的书&#xff0c;不如先自己动手写一个Windows程序。由于Windows程序的特有机制&#xff0c;不建议去写那种简单的HELLO WORLD&#x…

iOS 制作私有库framework + 图片资源的使用

1. 库的分类 开源库&#xff1a;公开源代码&#xff0c;能看到具体实现** 闭源库**&#xff1a;不公开源代码&#xff0c;是经过编译后的二进制文件&#xff0c;看不到具体实现&#xff1b;其中包括&#xff1a;静态库和动态库 2. 开源库的制作 我了解的开源库的制作&#xff0…

Python基础篇(一):如何使用PyCharm创建第一个Python项目(包含tools)

如何使用PyCharm创建第一个Python项目 前言1.创建Python项目2. 创建第一个python文件3. 编写运行第一个python程序4. 关于此工具的相关使用 前言 环境版本Python3.11.4PyCharm2023.1.2OSwindows10 PyCharm是一款由JetBrains开发的强大的Python集成开发环境&#xff08;IDE&am…

单个springboot整合rabbitmq

一、rabbitmq的搭建 centos7搭建rabbitmq:centos7安装rabbitmq_java-zh的博客-CSDN博客 二、在什么情况下选择rabbitMQ 常见的四种MQ比较 特 性ActiveMQRabbitMQRocketMQKafka语言JavaErlangJavaScala单机吞吐万万十万十万时效性msusmsms(以内)可用性高&#xff08;主从架构…

模拟电路系列分享-复杂阻容电路的频响

目录 概要 整体架构流程 技术名词解释 技术细节 1.基本变形 2.单元串联的粗略计算 3.推广结论 小结 概要 在基本单元电路的基础上&#xff0c;熟悉一些常见的变形电路&#xff0c;学会判断是高通还是低通&#xff0c;快速计算出截止频率是多少&#xff0c;对求解复杂电路的频率…

TypeScript ~ 掌握基本类型 ①

作者 : SYFStrive 博客首页 : HomePage &#x1f4dc;&#xff1a; TypeScript ~ TS &#x1f4cc;&#xff1a;个人社区&#xff08;欢迎大佬们加入&#xff09; &#x1f449;&#xff1a;社区链接&#x1f517; &#x1f4cc;&#xff1a;觉得文章不错可以点点关注 &…

MySQL高级sql语句操作二

MySQL高级sql语句操作二 一、EXISTS二、连接查询三、自我连接&#xff08;算排名&#xff09;四、CREATE VIEW&#xff08;视图&#xff09;五、UNION&#xff08;联集&#xff09;六、交集值七、无交集值八、CASE九、空值(NULL) 和 无值() 的区别十、正则表达式 一、EXISTS 用…

Kubernetes学习笔记-kubernetes应用扩展-自定义API对象(1)20230622

1、CustomResourceDefinitions介绍 开发者只需要只需向kubernetes api服务器提交CRD对象&#xff0c;即可定义新的资源类型。成功提交CRD之后&#xff0c;就能通过API服务器提交JSON清单或者YAML清单的方式创建自定义资源&#xff0c;以及其他kubernetes资源实例 创建一个CRD…

【雕爷学编程】Arduino动手做(118)---PS2接口模块

37款传感器与执行器的提法&#xff0c;在网络上广泛流传&#xff0c;其实Arduino能够兼容的传感器模块肯定是不止这37种的。鉴于本人手头积累了一些传感器和执行器模块&#xff0c;依照实践出真知&#xff08;一定要动手做&#xff09;的理念&#xff0c;以学习和交流为目的&am…