0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

LINUX中进程的最大理论数计算

科技绿洲 来源:Linux开发架构之路 作者:Linux开发架构之路 2023-11-09 09:17 次阅读

LINUX中进程的最大理论数计算

  • 每个进程都要在全局段描述表GDT中占据两个表项

每个进程的局部段描述表LDT都作为一个独立的段而存在,在全局段描述表GDT中要有一个表项指向这个段的起始地址,并说明该段的长度以及其他一些 参数。除上之外,每个进程还有一个TSS结构(任务状态段)也是一样。所以,每个进程都要在全局段描述表GDT中占据两个表项。

  • GDT的容量有多大呢?

寄存器中用作GDT表下标的位段宽度是13位,所以GDT中可以有213=8192213=8192个描述项。

除一些系统的开销(例如GDT中的第2项和第3项分别用于内核 的代码段和数据段,第4项和第5项永远用于当前进程的代码段和数据段,第1项永远是0,等等)以外,尚有8180个表项可供使用,所以理论上系统中最大的 进程数量是8180/2=40908180/2=4090。

所以系统中理论上最大的进程数是4090

系统中可创建的进程数实际值

linux内核通过进程标识值(process identification value)-PID来标示进程,PID是一个数,类型位pid_t, 实际上就是int类型

为了与老版本的Unix或者Linux兼容,PID的最大值默认设置位32768(short int 短整型的最大值)。

  • 查看

可以使用cat /proc/sys/kernel/pid_max来查看系统中可创建的进程数实际值

图片

  • 修改
ulimit -u 655351

设置完以后,虽然我们设置户创建进程数的硬限制和软限制都是65535,但是我们还不能使用创建65535个进程

我们在Linux还需要设置内核参数kernel.pid_max,这个参数我默认安装都是32768,

所以即使使用root帐户,却不设置这个内核参数,整个系统最多可以创建的进程数就是32768,所以我们需要进行如下设置:

sysctl -w kernel.pid_max=655351

最大线程数

linux 系统中单个进程的最大线程数有其最大的限制 PTHREAD_THREADS_MAX

这个限制可以在/usr/include/bits/local_lim.h中查看 对 linuxthreads 这个值一般是 1024,对于 nptl 则没有硬性的限制,仅仅受限于系统的资源

这个系统的资源主要就是线程的 stack 所占用的内存,用 ulimit -s 可以查看默认的线程栈大小,一般情况下,这个值是8M=8192KB

图片

可以写一段简单的代码验证最多可以创建多少个线程

include
#include #include void func()
{}int main(void)
{ int i = 0;
pthread_t thread; while ( 1 )
{ if (pthread_create(&thread, NULL, func, NULL) != 0)
{ return;
} i++; printf("i = %dn", i);
} return EXIT_SUCCESS;
}

图片

试验显示,在我们的系统(Ubuntu-14.04-LTS-64bit)中linuxthreads 上最多可以创建 381 个线程,之后就会返回 EAGAIN

LINUX中单个进程理论上可以创建的最大线程数

而32位系统中,可以穿件381个线程,这个值和理论完全相符,因为 32 位 linux 下的进程用户空间是 3G 的大小,也就是 3072M,用3072M/8M=3843072M/8M=384,但是实际上代码段和数据段等还要占用一些空间,这个值应该向下取整到 383,再减去主线程,得到 382。

那为什么 linuxthreads 上还要少一个线程呢?这可太对了,因为 linuxthreads 还需要一个管理线程

为了突破内存的限制,可以有两种方法

  • 用ulimit -s 1024减小默认的栈大小
  • 调用pthread_create的时候用pthread_attr_getstacksize设置一个较小的栈大小

要注意的是,即使这样的也无法突破1024 个线程的硬限制,除非重新编译 C 库

最大打开文件数

file-max系统最大打开文件描述符数

/proc/sys/fs/file-max中指定了系统范围内所有进程可打开的文件句柄的数量限制(系统级别, kernel-level).

The value in file-max denotes the maximum number of file handles that the Linux kernel will allocate).

当收到”Too many open files in system”这样的错误消息时, 就应该曾加这个值了.

对于2.2的内核, 还需要考虑inode-max, 一般inode-max设置为file-max的4倍. 对于2.4及以后的内核, 没有inode-max这个文件了.

查看实际值

可以使用cat /proc/sys/fs/file-max来查看当前系统中单进程可打开的文件描述符数目 186405

图片

设置

  • 临时性

echo 1000000 > /proc/sys/fs/file-max

  • 永久性:在/etc/sysctl.conf中设置

fs.file-max = 1000000

nr_open是单个进程可分配的最大文件数

内核支持的最大file handle数量,即一个进程最多使用的file handle数

the maximum number of files that can be opened by process。

A process cannot use more than NR_OPEN file descriptors.
一个进程不能使用超过NR_OPEN文件描述符。12

图片

nofile进程最大打开文件描述符数

查看实际值

ulimit -n

图片

当然默认查看的是软资源限制值soft limit,如果想要查看系统硬件所能支持的单进程最大打开文件描述符号的数目,可以使用ulimit -Hn

图片

设置

  • 临时性

通过ulimit -Sn设置最大打开文件描述符数的soft limit,注意soft limit不能大于hard limit(ulimit -Hn可查看hard limit)

另外ulimit -n默认查看的是soft limit,但是ulimit -n 1800000则是同时设置soft limit和hard limit。

对于非root用户只能设置比原来小的hard limit。

  • 永久性

上面的方法只是临时性的,注销重新登录就失效了,而且不能增大hard limit,只能在hard limit范围内修改soft limit。

若要使修改永久有效,则需要在/etc/security/limits.conf中进行设置(需要root权限),可添加如下两行,表示用户chanon最大打开文件描述符数的soft limit为1800000,hard limit为2000000。以下设置需要注销之后重新登录才能生效:

chanon soft nofile 102400
chanon hard nofile 40960012

设置nofile的hard limit还有一点要注意的就是hard limit不能大于/proc/sys/fs/nr_open,假如hard limit大于nr_open,注销后无法正常登录。

图片

可以修改nr_open的值:

echo 2000000 > /proc/sys/fs/nr_open

file-max, nr_open, onfile之间的关系

针对用户打开最大文件数的限制, 在limits.conf对应的nofile,不管是man手册还是文件中说明都只是一句话

“maximum number of open files”,

它其实对应是单个进程能打开的最大文件数,通常为了省事,我们想取消它的限制

根据man手册中,“values -1, unlimited or infinity indicating no limit”,-1、unlimited、infinity都是表明不做限制

可是当你实际给nofile设置成这个值,等你重启就会发现无法登录系统了。

由此可见,nofile是有一个上限的,同时用ulimit测试:

ulimit -n unlimited1

bash: ulimit: open files: cannot modify limit: 不允许的操作

图片

写一个简单的for循环得出:

for V in `seq 100000 10000000`;do ulimit -n $V;[[ $? != 0 ]]&&break;done1

再执行ulimit -n ,可以看到1048576就是nofile的最大值了,但为什么是这个值?

1024∗1024=10485761024∗1024=1048576,当然这并没有什么卵用。

再跟踪一下我们就会发现这个值其实是由内核参数nr_open定义的:

cat /proc/sys/fs/nr_open 1

图片

到此我们就要说起nr_open,与file-max了,网上在说到设置最大文件数时偶尔有些帖子也说到要修改file-max,字面上看file-max确实像是对应最大文件数,而在linux内核文档中它们两的解释是:

  • file-max:The value in file-max denotes the maximum number of file- handles that the Linux kernel will allocate. When you get lots of error messages about running out of file handles, you might want to increase this limit

执行:grep -r MemTotal /proc/meminfo | awk ‘{printf(“%d”,$2/10)}’,可以看到与file-max是相近的;

图片

  • nr_open:
    This denotes the maximum number of file-handles a process can allocate. Default value is 1024*1024 (1048576) which should be enough for most machines. Actual limit depends on RLIMIT_NOFILE resource limit.

file-handles(即文件句柄),然后相比而言在UNIX/LINUX中我们接触更多是file discriptor(FD,即文件描述符),似乎file-handle在windows中是一个类似file discrptor的东东,但是我们讨论的是linux,再google一下,我们可以精确到c语言中这两个概念的区别,

据他们的讨论file-handle应该是一个高层的对象,使用fopen,fread等函数来调用,而FD是底层的一个对象,可以通过open,read等函数来调用。

到此,我们应该可以下一个大致的结论了,file-max是内核可分配的最大文件数,nr_open是单个进程可分配的最大文件数,所以在我们使用ulimit或limits.conf来设置时,如果要超过默认的1048576值时需要先增大nr_open值(sysctl -w fs.nr_open=100000000或者直接写入sysctl.conf文件)。当然百万级别的单进程最大file-handle打开数应该也够用了吧。。

  1. 所有进程打开的文件描述符数不能超过/proc/sys/fs/file-max
  2. 单个进程打开的文件描述符数不能超过user limit中nofile的soft limit
  3. nofile的soft limit不能超过其hard limit
  4. nofile的hard limit不能超过/proc/sys/fs/nr_open

其他

2.4内核与2.6内核的主要区别

在2.4内核的典型系统上(AS3/RH9),线程是用轻量进程实现的,每个线程要占用一个进程ID,在服务器程序上,如果遇到高点击率访问,会造成进程表溢出,系统为了维护溢出的进程表,会有间歇的暂停服务现象,而2.6内核就不会发生由于大量线程的创建和销毁导致进程表溢出的问题

线程结束必须释放线程堆栈

就是说,线程函数必须调用pthread_exit()结束,否则直到主进程函数退出才释放,特别是2.6内核环境,线程创建速度飞快,一不小心立刻内存被吃光,这一点反倒是2.4内核环境好,因为2.4内核创建的是进程,而且线程创建速度比2.6内核慢几个数量级。特别提醒,在64位CPU,2.6内核创建线程的速度更加疯狂,要是太快的话,加上usleep ()暂停一点点时间比较好

不要编需要锁的线程应用

只有那些不需要互斥量的程序才能最大限度的利用线程编程带来的好处,否则只会更慢,2.6内核是抢占式内核,线程间共享冲突发生的几率远比2.4内核环境高,尤其要注意线程安全,否则就算是单CPU也会发生莫名其妙的内存不同步(CPU的高速缓存和主存内容不一致),Intel的新CPU为了性能使用NUMA架构,在线程编程中一定要注意扬长避短。

单进程服务器最大并发线程数与内存

很有趣,在默认的ulimit参数下,不修改内核头文件 AS3 512M内存最多1000并发持续连接 CentOS4.3 512M内存最多300并发持续连接 似乎是CentOS不如AS3,这里主要原因是ulimit的配置造成,两个系统默认的配置差距很大,要想单进程维持更多线程接收并发连接,就要尽量缩小 ulimit -s的参数,插更多的内存条,单进程服务器上2000并发一点都不难,POSIX默认的限制是每进程64线程,但NTPL并非纯正POSIX,不必理会这个限制,2.6内核下真正的限制是内存条的插槽数目(也许还有买内存的钱数) 最近几天的编程中,注意到在32位x86平台上2.6内核单进程创建最大线程数=VIRT上限/stack,与总内存数关系不大,32位x86系统默认的VIRT上限是3G(内存分配的3G+1G方式),默认 stack大小是10240K,因此单进程创建线程默认上限也就300(3072M / 10240K),用ulimit -s 修改stack到1024K则使上限升到大约3050。我手头没有64位系统,不知道2.6内核在64位上单进程创建线程上限(实际上是本人懒得在同事的机器上装fc4_x86_64)。前些天买了一套廉价的64位x86系统(64位赛杨+杂牌915主板),安装了CentOS4.3的x86_64版本,跑了一遍下面的小程序,得到的结果是:在ulimit -s 4096的情况下,单进程最大线程数在16000多一点,用top看 VIRT 的上限是64G,也就是36位, cat /proc/cpuinfo的结果是:address sizes : 36 bits physical, 48 bits virtual, 和我想象的标准64位系统不同, 我一直以为64位系统的内存空间也是64位的

附注1

单位里某BSD FANS用AMD64笔记本跑小程序测试线程创建速度(线程创建后立即phread_detach()然后紧跟着pthread_exit(),共计 100万个线程),同样源码OpenBSD竟然比FreeBSD快了3倍,什么时候OpenBSD也变得疯狂起来了?

附注2

测试单进程创建线程上限C源码(test.c)

#include
#include
#include #include #include void * thread_null(void);int main(int argc, char *argv[])
{ unsigned int i; int rc; pthread_t pool_id[65536]; //线程ID
sleep(1);
//创建线程 for(i = 0; i < 65536; i++)
{ rc = pthread_create(pool_id + i, 0, (void *)thread_null, NULL);
if (rc != 0)
{ fprintf(stderr, "pthread_create() failurernMax pthread num is %drn", i);
exit(-1);
} } fprintf(stdout, "Max pthread num is 65536rnYour system is power_fullrn");
exit(0);
}void * thread_null(void){ pthread_detach(pthread_self()); sleep(60);
pthread_exit(NULL);}
声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • Linux
    +关注

    关注

    87

    文章

    11285

    浏览量

    209264
  • PID
    PID
    +关注

    关注

    35

    文章

    1471

    浏览量

    85443
  • 代码
    +关注

    关注

    30

    文章

    4774

    浏览量

    68504
  • gdt
    gdt
    +关注

    关注

    0

    文章

    49

    浏览量

    10615
收藏 人收藏

    评论

    相关推荐

    Linux中进程和线程的深度对比

    关于进程和线程,在 Linux 中是一对儿很核心的概念。但是进程和线程到底有啥联系,又有啥区别,很多人还都没有搞清楚。
    发表于 10-14 16:47 1300次阅读
    <b class='flag-5'>Linux</b><b class='flag-5'>中进程</b>和线程的深度对比

    干货分享:基于嵌入式Linux中进程调度实现方法

    和网络协议。 1. 前言处理机(CPU)是整个计算机系统的核心资源,在多进程的操作系统中,进程往往多于处理机,这将导致各
    发表于 12-10 14:17

    Linux系统中进程如何查看及控制

    Linux系统中进程的查看及控制
    发表于 06-09 08:34

    WiFi5理论速率是如何计算出来的?

    率采用3/4,调制方式为64QAM调制,双流MIMO为2,因此最大理论速率为:84×2×6×3/4×100/0.5ms=151.2Mps。  TDD模式  蜂窝LTE工作在TDD模式下理论速率
    发表于 03-22 14:57

    Linux进程管理

    Linux进程管理 本章主要介绍进程的概念、状态、构成以及Linux进程的相关知识。 掌握进程
    发表于 04-28 14:57 0次下载

    Linux源码分析系列的进程

    概述:相关概念,进程在整个内核中的功能位置,源代码中进程相关的文件源代码中进程相关的文件。 Linux核心是多任务的,运行的程序称作进程 (
    发表于 11-03 22:26 32次下载

    Linux和UNIX可以用什么命令查看运行中进程的相关信息

      你可以使用ps命令。它能显示当前运行中进程的相关信息,包括进程的PID。Linux和UNIX都支持ps命令,显示所有运行中进程的相关信息。ps命令能提供一份当前
    发表于 01-20 09:42 6459次阅读

    Linux/Unix ulimit命令详解

    Linux对于每个用户,系统限制其最大进程。为提高性能,可以根据设备资源情况,设置各linux 用户的
    发表于 05-06 15:41 1137次阅读

    关于嵌入式Linux中进程调度的内容浅析

    处理机(CPU)是整个计算机系统的核心资源,在多进程的操作系统中,进程往往多于处理机,这将导致各进程
    发表于 04-03 09:02 855次阅读
    关于嵌入式<b class='flag-5'>Linux</b><b class='flag-5'>中进程</b>调度的内容浅析

    嵌入式Linux中进程调度怎样来解析

    处理机(CPU)是整个计算机系统的核心资源,在多进程的操作系统中,进程往往多于处理机,这将导致各进程
    发表于 11-06 11:54 715次阅读

    Linux进程间通信方式——管道

    管道是Linux中进程间通信的一种方式,它把一个程序的输出直接连接到另一个程序的输入。Linux的管道主要包括两种:无名管道和有名管道。
    发表于 06-01 09:13 1419次阅读
    <b class='flag-5'>Linux</b><b class='flag-5'>进程</b>间通信方式——管道

    虚拟机:linux 进程最大线程个数

    虚拟机:linux 进程最大线程个数
    的头像 发表于 06-22 15:56 2741次阅读
    虚拟机:<b class='flag-5'>linux</b> <b class='flag-5'>进程</b>的<b class='flag-5'>最大</b>线程个数

    深度剖析Linux中进程控制(上)

    Linux中,fork函数是非常重要的函数,它从已存在进程中创建一个新进程。新进程为子进程,而原进程
    的头像 发表于 05-12 10:49 537次阅读
    深度剖析<b class='flag-5'>Linux</b><b class='flag-5'>中进程</b>控制(上)

    深度剖析Linux中进程控制(下)

    Linux中,fork函数是非常重要的函数,它从已存在进程中创建一个新进程。新进程为子进程,而原进程
    的头像 发表于 05-12 10:49 495次阅读
    深度剖析<b class='flag-5'>Linux</b><b class='flag-5'>中进程</b>控制(下)

    Linux中进程、线程和协程的基础概念

    进程计算机中运行的程序的实例,它是操作系统中最基本的执行单元之一。每个进程都有自己的独立内存空间、系统资源和代码执行流。这意味着一个进程的崩溃通常不会影响其他
    的头像 发表于 12-06 09:22 836次阅读