Java的线程对象是存储在堆上的,所以,能够创建多少个线程,受到堆空间的大小限制,同时也受到每个线程的大小的限制,假如线程对象内部有一个非常大的数组字段,那就非常影响能够创建的线程的大小

我们的例子:

class Task implements Runnable {
//5M堆内存 128k的情况下,最多创建31个线程就挂了 byte[] bytes = new byte[1024 * 128]; @Override
public void run() {
while (true) {
try {
bytes[0] = 1;
Thread.sleep(10_100);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}

线程对象内部持有一个数组字段,这个数组的大小是128k,我们的测试代码:

public static void threadTest3() {
int count = 1;
try {
while (true) { Thread thread = new Thread(new Task());
thread.start();
System.out.println("==============================================" + count++);
}
} catch (Throwable e) {
System.out.println(count);
e.printStackTrace();
} }

运行测试的时候,我们将堆的内存设置的小一些:

-Xmx5m -Xms5m XX:+PrintGC -XX:-PrintGCDetails

输出的日志如下:

"C:\Program Files\Java\jdk1.8.0_152\bin\java.exe" -Xmx5m -Xms5m -XX:+PrintGC -XX:-PrintGCDetails  com.jiaoyiping.baseproject.concurrent.ThreadTest
[GC (Allocation Failure) 1024K->732K(5632K), 0.0006644 secs]
[GC (Allocation Failure) 1746K->1023K(5632K), 0.0012964 secs]
[GC (Allocation Failure) 1818K->1417K(5632K), 0.0005073 secs]
[GC (Allocation Failure) 2441K->1546K(5632K), 0.0008099 secs]
[GC (Allocation Failure) 2570K->1773K(5632K), 0.0009678 secs]
==============================================1
[GC (Allocation Failure) 2752K->2046K(5632K), 0.0012278 secs]
==============================================2
==============================================3
==============================================4
==============================================5
==============================================6
==============================================7
==============================================8
[GC (Allocation Failure) 2962K->3022K(5632K), 0.0007815 secs]
==============================================9
==============================================10
==============================================11
==============================================12
==============================================13
==============================================14
==============================================15
[GC (Allocation Failure) 3938K->3950K(5632K), 0.0005197 secs]
[Full GC (Ergonomics) 3950K->3132K(5632K), 0.0098480 secs]
==============================================16
==============================================17
==============================================18
==============================================19
==============================================20
==============================================21
==============================================22
[GC (Allocation Failure) 4090K->4220K(5632K), 0.0004522 secs]
[Full GC (Ergonomics) 4220K->3975K(5632K), 0.0085720 secs]
==============================================23
==============================================24
==============================================25
==============================================26
==============================================27
==============================================28
==============================================29
[Full GC (Ergonomics) 4920K->4876K(5632K), 0.0110795 secs]
[Full GC (Allocation Failure) 4876K->4856K(5632K), 0.0086577 secs]
==============================================30
[Full GC (Ergonomics) 5015K->4985K(5632K), 0.0025122 secs]
[Full GC (Allocation Failure) 4985K->4985K(5632K), 0.0022044 secs]
31
java.lang.OutOfMemoryError: Java heap space
at com.jiaoyiping.baseproject.concurrent.Task.<init>(ThreadTest.java:105)
at com.jiaoyiping.baseproject.concurrent.ThreadTest.threadTest3(ThreadTest.java:74)
at com.jiaoyiping.baseproject.concurrent.ThreadTest.main(ThreadTest.java:17)
[Full GC (Ergonomics) 5118K->4995K(5632K), 0.0108311 secs]
[Full GC (Ergonomics) 5107K->5029K(5632K), 0.0034520 secs]
[Full GC (Ergonomics) 5107K->5058K(5632K), 0.0034841 secs]
[Full GC (Ergonomics) 5107K->5064K(5632K), 0.0032645 secs]
[Full GC (Ergonomics) 5107K->5097K(5632K), 0.0034425 secs]
[Full GC (Ergonomics) 5107K->5107K(5632K), 0.0030512 secs]
[Full GC (Ergonomics) 5107K->5107K(5632K), 0.0027000 secs]
[Full GC (Ergonomics) 5109K->5109K(5632K), 0.0026329 secs]
[Full GC (Ergonomics) 5111K->5111K(5632K), 0.0026942 secs]
[Full GC (Ergonomics) 5113K->5113K(5632K), 0.0024612 secs]
[Full GC (Ergonomics) 5116K->5116K(5632K), 0.0028061 secs]
[Full GC (Ergonomics) 5118K->5118K(5632K), 0.0027730 secs]
[Full GC (Ergonomics) 5119K->5119K(5632K), 0.0039374 secs]
[Full GC (Allocation Failure) 5119K->5118K(5632K), 0.0037878 secs]
[Full GC (Ergonomics) 5119K->5119K(5632K), 0.0031486 secs]
[Full GC (Allocation Failure) 5119K->5119K(5632K), 0.0028138 secs]

当创建了30个线程的时候,就出现了堆内存的溢出

如果将bytes的值改变为56K呢,可以创建70个线程

-Xss影响的是栈的深度(比如栈的深度大的话,能够递归调用的次数就越多)

我们构造出来递归调用的例子,用来测试-Xss参数对递归调用的次数的影响

 public static void threadTest4() {
try {
add(0);
} catch (Throwable throwable) {
throwable.printStackTrace();
}
} static int add(int count) {
System.out.println("invoke count" + (count + 1));
return add(count + 1);
}

当-Xss设置为128k的时候:

当-Xss设置为256K的时候:

jvm推荐的-Xss设置最少是108k,小于108K的时候,结果是不可预料的

5k-64k 无法创建虚拟机:

4k和以下很大的数值:

64K- 108K

这个值默认是1M我们一般不需要修改,值需要知道-Xss和栈的深度有关即可,通过以上的例子我们得到以下结论:堆内存的大小,和线程对象本身的大小,影响能够创建的线程的个数,-Xss影响栈的深度

除了Java虚拟机之外,操作系统也会对能创建多少个进程和线程进行限制

通过两个文件我们可以查看和修改这些限制数:

最大的PID

/proc/sys/kernel/threads-max

最多的线程数:

/proc/sys/kernel/pid_max

Java中的堆内存设置对线程创建数的影响以及-Xss参数的记录的更多相关文章

  1. Java基础-Java中的堆内存和离堆内存机制

    Java基础-Java中的堆内存和离堆内存机制 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任.

  2. Java中的堆内存、栈内存、静态存储区

    一.栈 栈的优势是,存取速度比堆要快,仅次于直接位于CPU中的寄存器,当超过变量的作用域后,java会自动释放掉为该变量分配的内存空间,该内存空间可以立刻被另作他用.但缺点是,存在栈中的数据大小与生存 ...

  3. java中的堆内存和栈内存

    Java把内存分成两种: 一种叫做栈内存 一种叫做堆内存 栈内存 : 在函数中定义的一些基本类型的变量和对象的引用变量都是在函数的栈内存中分配.当在一段代码块中定义一个变量时,java就在栈中为这个变 ...

  4. java中的堆与栈

    Java 中的堆和栈 Java把内存划分成两种:一种是栈内存,一种是堆内存. 在函数中定义的一些基本类型的变量和对象的引用变量都在函数的栈内存中分配 . 当在一段代码块定义一个变量时,Java就在栈中 ...

  5. Java堆内存设置

    转自:https://blog.csdn.net/Qiuzhongweiwei/article/details/81023645 堆内存设置 原理 JVM堆内存分为2块:永久空间和堆空间. 永久即持久 ...

  6. 【转】JVM 堆内存设置原理

    堆内存设置 原理 JVM堆内存分为2块:Permanent Space 和 Heap Space. Permanent 即 持久代(Permanent Generation),主要存放的是Java类定 ...

  7. JVM堆内存设置

    今天碰到了一个题目,讲的是关于堆内存的问题,题目如下   下面哪种情况会导致持久区jvm堆内存溢出? A.循环上万次的字符串处理 B.在一段代码内申请上百M甚至上G的内存 C.使用CGLib技术直接操 ...

  8. JDK8中JVM堆内存划分

    一:JVM中内存 JVM中内存通常划分为两个部分,分别为堆内存与栈内存,栈内存主要用运行线程方法 存放本地暂时变量与线程中方法运行时候须要的引用对象地址. JVM全部的对象信息都 存放在堆内存中.相比 ...

  9. [转]JVM 堆内存设置原理

    堆内存设置 原理 JVM堆内存分为2块:Permanent Space 和 Heap Space. Permanent 即 持久代(Permanent Generation),主要存放的是Java类定 ...

随机推荐

  1. [Asp.net core]使用ssh命令发布asp.net core项目

    命令 # 移除之前发布的包 rm -rf ./.Publish rm -rf ./Wolfy.Blog.tar.gz # 编译并发布 将发布包打包在.Publish目录下 -o "../.P ...

  2. 对指针和引用的理解(c++)

    1.指针 typedef说明一种新类型名,来代替已有类型名. a.案例:typedef char* String_t和#define String_d char *这两句在使用上的区别? 1)前者声明 ...

  3. Python调试打印错误信息

    try: ..... except Exception, e: print 'repr(e):\t', repr(e)

  4. EAS开发之挂菜单

        一:以管理员账号登录   二:挂菜单     点击菜单栏"系统"——客户化菜单编辑——选中上级目录——点击 新建——命名.键入唯一编码,把ui.java类的全路径,拷贝到 ...

  5. 【PMP】项目整合管理

    项目整合的七个过程组: 制定项目章程 制定项目管理计划 指导语管理项目工作 管理项目知识 监控项目工作 实施整体变更控制 结束项目或阶段 1.制定项目章程 1.1 定义 制定项目章程是编写一份正式批准 ...

  6. Rabbit五种消息队列学习(二) – 简单队列

    队列结构图 P:消息的生产者 C:消息的消费者 红色:队列 生产者将消息发送到队列,消费者从队列中获取消息. 测试 1.连接MQ public static Connection getConnect ...

  7. 分析轮子(二)- << ,>>,>> (左移、右移、无符号右移)

    前言:写 分析轮子(一)-ArrayList.java 的时候看到源码中有 int newCapacity = oldCapacity + (oldCapacity >> 1); 这样的代 ...

  8. Android ANR的产生与分析

      ANR即Application Not Responding应用无响应,一般在ANR的时候会弹出一个应用无响应对话框.也许有些开发者在使用某些手机开发中不在弹出应用无响应弹出框,特别是国产手机An ...

  9. maven仓库中心mirrors配置多个下载中心(执行最快的镜像)

    E:\Program FilesApache Software Foundationapache-maven-3.5.4-binconf\settings.xmlmaven仓库中心mirrors配置多 ...

  10. ThreadPoolExcutor 线程池 异常处理 (上篇)

    前言 最近看到crossoverJie的一篇文章:一个线程罢工的诡异事件 首先感谢原作者的分享,自己获益匪浅.然后是回想到自己的一次面试经历,面试官提问了线程池中的线程出现了异常该怎样捕获?会导致什么 ...