`

内存溢出,无法创建新的本地线程的原因和解决方案,java.lang.OutOfMemoryError: unable to create new native

    博客分类:
  • JVM
阅读更多

内存溢出,无法创建新的本地线程的原因和解决方案,java.lang.OutOfMemoryError: unable to create new native thread

 

 

I recently came across this exception on a couple of java systems that use many threads java.lang.OutOfMemoryError: unable to create new native thread. The strange thing was that the JVM had been assigned a lot of memory (1.5GB) and that it had at least half the memory available. Michele found this article that points out that the more memory you give to the JVM the more likely you are to get java.lang.OutOfMemoryError: unable to create new native thread exceptions when you have many threads.

JVM已经分配了足够多的内存了(1.5GB),而且至少有一半的可用内存。但是当你有很多的线程在运行时,你给JVM的内存越多,你越容易出现无法创建本地线程的内存溢出。


Which makes perfect sense when you think about it. Each 32 bit process on Windows has 2GB "available" memory as 2GB is reserved to Windows. In my case the JVM grabbed 1.5 GB leaving 500MB. Part of the 500MB was used to map system dlls etc in memory so less than 400 MB was left. Now to the crucial point: When you create a thread in java it creates a Thread object in the JVM memory but it also creates a operating system thread. The operating system creates the thread with a thread stack in the 400MB that is left, not in the 1.5 GB allocated in the JVM. Java 1.4 uses a default stack size of 256kb but Java 1.5 uses a 1MB stack per thread. So, in the 400MB left to process I could only generate ~400 threads. Absurd but true: to create more threads you have to reduce the memory allocated to the JVM. Another option is to host the JVM in your own process using JNI.
在32位的Windows机器上,最大有2G的内存,所以JVM一般使用1.5G留下500M。其中的一部分用来映射系统的dll到内存里面,所以只剩下了400M.此时,我们创建了一个java线程,它不仅在JVM的内存创建了线程对象,同时创建了操作系统的线程。
操作系统创建线程,使用了400M里的线程堆栈而不是JVM里面1.5GB的。 每个线程,Java 1.4默认使用 256k, 1.5默认使用 1M的堆栈。那么 400MB的内存,最多可以创建400个线程。为了创建更多的线程,你必须减少你的JVM分配的内存数。

This formula gives a decent estimate for the number of threads you can create:
(MaxProcessMemory - JVMMemory - ReservedOsMemory) / (ThreadStackSize) = Number of threads

For Java 1.5 I get the following results assuming that the OS reserves about 120MB:
1.5GB allocated to JVM: (2GB-1.5Gb-120MB)/(1MB) = ~380 threads
1.0GB allocated to JVM: (2GB-1.0Gb-120MB)/(1MB) = ~880 threads

Java 1.4 uses 256kb for the thread stack which lets you create a lot more threads:
1.5GB allocated to JVM: ~1520 threads
1.0GB allocated to JVM: ~3520 threads

I have not tried the 3GB switch but it should in theory let you create more threads.

分享到:
评论

相关推荐

Global site tag (gtag.js) - Google Analytics