Java虚拟机最多支持多少个线程?

做者:miracle1919
http://www.importnew.com/1078...

McGovernTheory在StackOverflow提了这样一个问题:Java虚拟机最多支持多少个线程?跟虚拟机开发商有关么?跟操做系统呢?还有其余的因素吗?html

Eddie的回答:java

这取决于你使用的CPU,操做系统,其余进程正在作的事情,你使用的Java的版本,还有其余的因素。我曾经见过一台Windows服务器在宕机以前有超过6500个线程。固然,大多数线程什么事情也没有作。一旦一台机器上有差很少6500个线程(Java里面),机器就会开始出问题,并变得不稳定。面试

以个人经验来看,JVM容纳的线程与计算机自己性能是正相关的。后端

固然了,你要有足够的本机内存,而且给Java分配了足够的内存,让每一个线程均可以拥有栈(虚拟机栈),能够作任何想作的事情。任何一台拥有现代CPU(AMD或者是Intel最近的几代)和1-2G内存(取决于操做系统)的机器很容易就能够支持有上千个线程的Java虚拟机。服务器

若是你须要一个更精确的答案,最好是本身作压测。多线程

Charlie Martin的回答:架构

这里有不少的参数(能够设置)。对于特定的虚拟机,都会有本身的运行时参数。(最大线程数)必定程度上由操做系统决定的:底层的操做系统要给线程提供哪些支持?施加哪些限制?虚拟机使用的是原生的操做系统的线程仍是red thread或者green thread?函数

操做系统提供的支持是另外一个问题。若是你向下面这样写Java程序:工具

class DieLikeADog {
      public static void main(String[] argv){
          for(;;){
             new Thread(new SomeRunaable).start();
          }
      }
 }

(不要抱怨语法细节,这才刚刚开始)那你固然但愿能获得成百上千个运行的线程。可是,建立一个线程的成本是相对较大的,(过多线程)调度的开销会变得突出。可否让这些线程作有用的事情还不肯定。性能

升级版

好了,火烧眉毛了!下面是个人一个加了点润色的小的测试程序:

public class DieLikeADog {
    private static Object s = new Object();
    private static int count = 0;
    public static void main(String[] argv){
        for(;;){
            new Thread(new Runnable(){
                    public void run(){
                        synchronized(s){
                            count += 1;
                            System.err.println("New thread #"+count);
                        }
                        for(;;){
                            try {
                                Thread.sleep(1000);
                            } catch (Exception e){
                                System.err.println(e);
                            }
                        }
                    }
                }).start();
        }
    }
}

在Intel的OS/X 10.5.6系统上,Java 5的输出以下:

New thread #2547
New thread #2548
New thread #2549
Can't create thread: 5
New thread #2550
Exception in thread "main" java.lang.OutOfMemoryError: unable to create new native thread
at java.lang.Thread.start0(Native Method)
at java.lang.Thread.start(Thread.java:592)
at DieLikeADog.main(DieLikeADog.java:6)

benjismith的回答:

读了Charlie Martin的回复之后,我很想知道堆内存的大小是否可以给建立的线程数带来不一样,而后我就被结果惊呆了:在Vista Home Premium SP1系统上,使用JDK 1.6.0_11,设置堆内存的大小从2M到1024M来执行Charlie的测试程序。好比:建立2M的堆内存,我使用的虚拟机参数是:-Xms2m -Xmx2m.

下面是个人测试结果:

2 mb --> 5744 threads
4 mb --> 5743 threads
8 mb --> 5735 threads
12 mb --> 5724 threads
16 mb --> 5712 threads
24 mb --> 5687 threads
32 mb --> 5662 threads
48 mb --> 5610 threads
64 mb --> 5561 threads
96 mb --> 5457 threads
128 mb --> 5357 threads
192 mb --> 5190 threads
256 mb --> 5014 threads
384 mb --> 4606 threads
512 mb --> 4202 threads
768 mb --> 3388 threads
1024 mb --> 2583 threads

因此,堆的大小确实很重要。可是,堆大小和最大线程数倒是呈反比例关系。

这太诡异了!

Neil Coffey的回答:

绝对理论上的最大线程数是进程的用户地址空间除以线程栈的大小(现实中,若是内存所有给线程栈使用,就不会有能运行的程序了)。所以,以32位Windows系统为例,每个进程的用户地址空间是2G,假如每一个线程栈的大小是128K,最多会有16384(=2*1024*1024 / 128)个线程。实际在XP系统上,我发现大约能启动13000个线程。

而后,我认为,你的问题本质上是:(a)你是否能够在你的代码中有效的管理许多的线程,不让他们作很显然是愚蠢的事情(好比:让他们在同一个object对象上等待随后被调用notifyAll()…),(b)操做系统是否能够有效地管理这许多线程。基本上来讲,若是(a)的答案是”yes”的话,(b)的答案也是”yes”。

很巧的是,你能够在Thread的构造函数中设置线程栈的大小,可是,你不须要也不该该把这个和虚拟机参数弄混淆。

关注公众号Java技术栈回复"面试"获取我整理的2020最全面试题及答案。

推荐去个人博客阅读更多:

1.Java JVM、集合、多线程、新特性系列教程

2.Spring MVC、Spring Boot、Spring Cloud 系列教程

3.Maven、Git、Eclipse、Intellij IDEA 系列工具教程

4.Java、后端、架构、阿里巴巴等大厂最新面试题

以为不错,别忘了点赞+转发哦!

相关文章
相关标签/搜索