热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

突破FD_SETSIZE限制的连接数之boost库的asio的中fd_set的适配器的设计

前言这个也我在公司的使用自己写库clib中在win上连接服务器时中只能连接63个的客户端,这个我一开始还以为是自己的配置文件中配置问题呢就没有注意这个问题。直到最近

前言

这个也我在公司的使用自己写库clib中在win上连接服务器时中只能连接63个的客户端,这个我一开始还以为是自己的配置文件中配置问题呢 就没有注意这个问题。 直到最近自己服务器上业务写完了, 需要压力测试时, 这个问题我可查有几天, 一开始我以为是自己编码中写死了呢, 为什么这样说呢! 在我的知识体系中select最大连接数是1024, 我找很长时间 最后在发现在win 上 FD_SETSIZE的宏定义是64。 这才定位到问题。

使用原生的socket写 服务器或者写客户端都会使用监听文件描述符 读写的状态, 都遇到文件描述符到1024或者64的连接限制, 但是linux内核2.6以后有了epoll就可以完美突破这个限制, 当是在win上和其它平台没有epoll支持, 就要使用select 和poll等等 函数,它们都是连接数 的限制的。

还有一次我面试一家游戏公司, 面试官问到我知道linux上最大连接数, 我当时以为是 一个命令呢, ulimit修改linux连接数呢,现在我了解一点 连接数关于编程方面的限制。与大家分享一下


正文


一, 简单介绍 fd_set使用


1, fd_set 结构中定义结构体的大小

typedef struct fd_set {u_int fd_count; /* how many are SET? */SOCKET fd_array[FD_SETSIZE]; /* an array of SOCKETs */// FD_SETSIZE: 最大连接数据
} fd_set;

2, 添加文件描述符到 fd_set中的数组中fd_array

#define FD_SET(fd, set) do { \u_int __i; \for (__i &#61; 0; __i <((fd_set FAR *)(set))->fd_count; __i&#43;&#43;) { \ // 查找fd_array数组中文件描述符的位置if (((fd_set FAR *)(set))->fd_array[__i] &#61;&#61; (fd)) { \break; \} \} \if (__i &#61;&#61; ((fd_set FAR *)(set))->fd_count) { \if (((fd_set FAR *)(set))->fd_count fd_array[__i] &#61; (fd); \((fd_set FAR *)(set))->fd_count&#43;&#43;; \} \} \
} while(0, 0)

3, 在数组中删除文件描述符

#define FD_CLR(fd, set) do { \u_int __i; \for (__i &#61; 0; __i <((fd_set FAR *)(set))->fd_count ; __i&#43;&#43;) { \if (((fd_set FAR *)(set))->fd_array[__i] &#61;&#61; fd) { \while (__i <((fd_set FAR *)(set))->fd_count-1) { \((fd_set FAR *)(set))->fd_array[__i] &#61; \((fd_set FAR *)(set))->fd_array[__i&#43;1]; \__i&#43;&#43;; \} \((fd_set FAR *)(set))->fd_count--; \break; \} \} \
} while(0, 0)

extern int PASCAL FAR __WSAFDIsSet(SOCKET fd, fd_set FAR *);#define FD_ZERO(set) (((fd_set FAR *)(set))->fd_count&#61;0)#define FD_ISSET(fd, set) __WSAFDIsSet((SOCKET)(fd), (fd_set FAR *)(set))

如果你还是不了解可以看我以前写文章地址&#xff1a;https://chensongpoixs.github.io/2017/10/31/Linux网络编程之IO复用(select函数的使用)/


二, 分析问题怎么突破FD_SETSIZE的限制


1, FD_SETSIZE的win宏定义是64

我们只要自己定义 fd_set 的结构 数组的大小就可以了

这里我们就要使用适配器(fd_set_adapter)怎么一个东西 , 我相信写android的界面的知道怎么一个东西 &#xff0c; 我也前几天写android东西&#xff0c;才想起以前写android一些东西&#xff0c; 发现android一些设计模式太多了&#xff0c; 适合我们学习的设计模式的&#xff0c;挺搞笑的 我连java的基本List数据结构都不会使用&#xff0c; 我以为跟C&#43;&#43;的数据结构一样的使用方式呢&#xff0c; 足足花了我半天的时间&#xff0c;我才知道要 new一下才可以使用&#xff0c; 要给这个对象申请内存空间。 毕竟android系统是google开源出来了&#xff0c; 里面有很多东西要学习的

这个适配器我也看boost中源码有适配器对win做了适配器&#xff0c;boost库也是C&#43;&#43;中著名网络库&#xff0c;

我是借鉴boost1.6是动态增加数组

namespace chen {class cfd_set_adapter{private:enum { default_fd_set_size &#61; 1024 };struct _fd_set {uint32 m_count; //typedef unsigned int u_int;socket_type m_fd_array[1];//动态申请数组大小};public:explicit cfd_set_adapter();~cfd_set_adapter();public:bool init();void destroy();public:bool set(socket_type descriptor);// FD_ISSETbool is_set(socket_type descriptor) const;// []operator fd_set*();void reset();socket_type max_descriptor() const{return m_max_descriptor_fd;}private://扩容void reserve(uint32 size);private:cfd_set_adapter(const cfd_set_adapter&);cfd_set_adapter& operator&#61;(const cfd_set_adapter&);private:_fd_set* m_fd_set;uint32 m_capacity;// 数组的大小uint32 m_max_descriptor_fd;};
} // namespace chen

2, 分析boost 库中怎么使用select 适配器的fd_set_adapter

使用在主要在run方法中

主要思想是 文件描述符队列 op_queue &#xff08;连接上文件描述符 &#xff09;

fd_sets_是分别放read,write和except三个数组

然后把队列中文件描述符放到fd_set_三个文件描述符数组中 在放到select函数中

void select_reactor::run(bool block, op_queue& ops)
{boost::asio::detail::mutex::scoped_lock lock(mutex_);#if defined(BOOST_ASIO_HAS_IOCP)// Check if the thread is supposed to stop.if (stop_thread_)return;
#endif // defined(BOOST_ASIO_HAS_IOCP)// Set up the descriptor sets.for (int i &#61; 0; i max_fd)max_fd &#61; fd_sets_[i].max_descriptor();}#if defined(BOOST_ASIO_WINDOWS) || defined(__CYGWIN__)// Connection operations on Windows use both except and write fd_sets.have_work_to_do &#61; have_work_to_do || !op_queue_[connect_op].empty();fd_sets_[write_op].set(op_queue_[connect_op], ops);if (fd_sets_[write_op].max_descriptor() > max_fd)max_fd &#61; fd_sets_[write_op].max_descriptor();fd_sets_[except_op].set(op_queue_[connect_op], ops);if (fd_sets_[except_op].max_descriptor() > max_fd)max_fd &#61; fd_sets_[except_op].max_descriptor();
#endif // defined(BOOST_ASIO_WINDOWS) || defined(__CYGWIN__)// We can return immediately if there&#39;s no work to do and the reactor is// not supposed to block.if (!block && !have_work_to_do)return;// Determine how long to block while waiting for events.timeval tv_buf &#61; { 0, 0 };timeval* tv &#61; block ? get_timeout(tv_buf) : &tv_buf;lock.unlock();// Block on the select call until descriptors become ready.boost::system::error_code ec;int retval &#61; socket_ops::select(static_cast(max_fd &#43; 1),fd_sets_[read_op], fd_sets_[write_op], fd_sets_[except_op], tv, ec);// Reset the interrupter.if (retval > 0 && fd_sets_[read_op].is_set(interrupter_.read_descriptor())){interrupter_.reset();--retval;}lock.lock();// Dispatch all ready operations.if (retval > 0){
#if defined(BOOST_ASIO_WINDOWS) || defined(__CYGWIN__)// Connection operations on Windows use both except and write fd_sets.fd_sets_[except_op].perform(op_queue_[connect_op], ops);fd_sets_[write_op].perform(op_queue_[connect_op], ops);
#endif // defined(BOOST_ASIO_WINDOWS) || defined(__CYGWIN__)// Exception operations must be processed first to ensure that any// out-of-band data is read before normal data.for (int i &#61; max_select_ops - 1; i >&#61; 0; --i)fd_sets_[i].perform(op_queue_[i], ops);}timer_queues_.get_ready_timers(ops);
}

三, 测试效果图

因为在win限制是64 我使用200使用200 机器人连接gateway gateway的session回话id 从1000 到800 机器人每30秒发送一个数据包&#xff0c; 下面的测试流程图

源码路径

https://github.com/chensongpoixs/clib


结语

个人博客地址:https://chensongpoixs.github.io/


推荐阅读
  • 本文介绍了在开发Android新闻App时,搭建本地服务器的步骤。通过使用XAMPP软件,可以一键式搭建起开发环境,包括Apache、MySQL、PHP、PERL。在本地服务器上新建数据库和表,并设置相应的属性。最后,给出了创建new表的SQL语句。这个教程适合初学者参考。 ... [详细]
  • 目录实现效果:实现环境实现方法一:基本思路主要代码JavaScript代码总结方法二主要代码总结方法三基本思路主要代码JavaScriptHTML总结实 ... [详细]
  • android listview OnItemClickListener失效原因
    最近在做listview时发现OnItemClickListener失效的问题,经过查找发现是因为button的原因。不仅listitem中存在button会影响OnItemClickListener事件的失效,还会导致单击后listview每个item的背景改变,使得item中的所有有关焦点的事件都失效。本文给出了一个范例来说明这种情况,并提供了解决方法。 ... [详细]
  • baresip android编译、运行教程1语音通话
    本文介绍了如何在安卓平台上编译和运行baresip android,包括下载相关的sdk和ndk,修改ndk路径和输出目录,以及创建一个c++的安卓工程并将目录考到cpp下。详细步骤可参考给出的链接和文档。 ... [详细]
  • javascript  – 概述在Firefox上无法正常工作
    我试图提出一些自定义大纲,以达到一些Web可访问性建议.但我不能用Firefox制作.这就是它在Chrome上的外观:而那个图标实际上是一个锚点.在Firefox上,它只概述了整个 ... [详细]
  • XML介绍与使用的概述及标签规则
    本文介绍了XML的基本概念和用途,包括XML的可扩展性和标签的自定义特性。同时还详细解释了XML标签的规则,包括标签的尖括号和合法标识符的组成,标签必须成对出现的原则以及特殊标签的使用方法。通过本文的阅读,读者可以对XML的基本知识有一个全面的了解。 ... [详细]
  • Nginx使用AWStats日志分析的步骤及注意事项
    本文介绍了在Centos7操作系统上使用Nginx和AWStats进行日志分析的步骤和注意事项。通过AWStats可以统计网站的访问量、IP地址、操作系统、浏览器等信息,并提供精确到每月、每日、每小时的数据。在部署AWStats之前需要确认服务器上已经安装了Perl环境,并进行DNS解析。 ... [详细]
  • 本文介绍了Python高级网络编程及TCP/IP协议簇的OSI七层模型。首先简单介绍了七层模型的各层及其封装解封装过程。然后讨论了程序开发中涉及到的网络通信内容,主要包括TCP协议、UDP协议和IPV4协议。最后还介绍了socket编程、聊天socket实现、远程执行命令、上传文件、socketserver及其源码分析等相关内容。 ... [详细]
  • 在Android开发中,使用Picasso库可以实现对网络图片的等比例缩放。本文介绍了使用Picasso库进行图片缩放的方法,并提供了具体的代码实现。通过获取图片的宽高,计算目标宽度和高度,并创建新图实现等比例缩放。 ... [详细]
  • Nginx使用(server参数配置)
    本文介绍了Nginx的使用,重点讲解了server参数配置,包括端口号、主机名、根目录等内容。同时,还介绍了Nginx的反向代理功能。 ... [详细]
  • 本文介绍了如何使用php限制数据库插入的条数并显示每次插入数据库之间的数据数目,以及避免重复提交的方法。同时还介绍了如何限制某一个数据库用户的并发连接数,以及设置数据库的连接数和连接超时时间的方法。最后提供了一些关于浏览器在线用户数和数据库连接数量比例的参考值。 ... [详细]
  • 原文地址:https:www.cnblogs.combaoyipSpringBoot_YML.html1.在springboot中,有两种配置文件,一种 ... [详细]
  • 【Windows】实现微信双开或多开的方法及步骤详解
    本文介绍了在Windows系统下实现微信双开或多开的方法,通过安装微信电脑版、复制微信程序启动路径、修改文本文件为bat文件等步骤,实现同时登录两个或多个微信的效果。相比于使用虚拟机的方法,本方法更简单易行,适用于任何电脑,并且不会消耗过多系统资源。详细步骤和原理解释请参考本文内容。 ... [详细]
  • Android Studio Bumblebee | 2021.1.1(大黄蜂版本使用介绍)
    本文介绍了Android Studio Bumblebee | 2021.1.1(大黄蜂版本)的使用方法和相关知识,包括Gradle的介绍、设备管理器的配置、无线调试、新版本问题等内容。同时还提供了更新版本的下载地址和启动页面截图。 ... [详细]
  • 图解redis的持久化存储机制RDB和AOF的原理和优缺点
    本文通过图解的方式介绍了redis的持久化存储机制RDB和AOF的原理和优缺点。RDB是将redis内存中的数据保存为快照文件,恢复速度较快但不支持拉链式快照。AOF是将操作日志保存到磁盘,实时存储数据但恢复速度较慢。文章详细分析了两种机制的优缺点,帮助读者更好地理解redis的持久化存储策略。 ... [详细]
author-avatar
永恒多一天_313
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有