JVM中的堆和非堆内存

Posted messpro

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了JVM中的堆和非堆内存相关的知识,希望对你有一定的参考价值。

JVM内存包含以下部分:

  • 堆内存,它是Java对象的存储
  • 非堆内存,Java用于存储加载的类和其他元数据
  • 其他,JVM代码本身,JVM内部结构,加载的探查器代理代码和数据等。

JVM有一个堆,它是运行时数据区,从中分配所有类实例和数组的内存。它是在JVM启动时创建的。

可以使用以下VM选项配置堆大小:

1

2

-Xmx<size> - to set the maximum Java heap size

-Xms<size> - to set the initial Java heap size

默认情况下,最大堆大小为64 Mb。

对象的堆内存由自动内存管理系统回收,该系统称为垃圾收集器。堆可以是固定大小,也可以扩展和缩小,具体取决于垃圾收集器的策略。

非堆

此外,JVM还有堆以外的内存,称为非堆内存。它在JVM启动时创建并存储每类结构,例如运行时常量池,字段和方法数据,方法和构造函数的代码,以及实习字符串。

不幸的是,JVM在非堆内存上提供的唯一信息是它的整体大小。没有关于非堆内存内容的详细信息。

非堆内存大小的异常增长可能表示存在潜在问题,在这种情况下,您可以检查以下内容:

  • 如果存在类加载问题,例如泄漏的加载器。
  • 如果有字符串被大量实施。为了检测这种问题,可以使用对象分配记录。

如果应用程序确实需要大量非堆内存且默认最大大小为64 Mb是不够的,则可以借助-XX:MaxPermSizeVM选项扩大最大大小。例如,-XX:MaxPermSize=128m设置128 Mb的大小。

 

 

 

 

堆栈和堆之间的异同

Similarities and Differences Between Stack and Heap

两者都是Java分配内存的方式,两者都存储在RAM中。但是,为了使事情更容易记住,堆用于动态内存分配,而堆栈用于静态分配。

它存放在哪里? 在堆栈上分配的变量可以直接从内存访问,因此,这些变量可以非常快速地运行。另一方面,访问堆上的对象需要更多时间。

分配何时发生? 在堆栈上,编译程序时会发生内存分配。同时,在堆上,它在程序运行时开始。

既然如此,那么在编译之前,如果要使用堆栈,则需要知道需要多少数据和内存。堆栈的另一个限制是它无法处理需要大量内存的大块变量。如果您不知道在运行时需要多少数据,或者您需要大量数据的内存,那么您需要使用堆。

简而言之…

堆栈stack

  • 堆栈的大小将随着方法和函数根据需要创建和删除局部变量而变化。
  • 分配内存然后随后释放,而无需管理内存分配。
  • 堆栈的大小限制,可能会根据您使用的操作系统而有所不同。
  • 只要创建它们的函数正在运行,就会存在存储在堆栈中的变量。

堆heap

  • 内存不是自动管理的,也不是由管理堆栈的方式由中央处理单元严格管理的。当不再需要这些块时,您需要自己释放已分配的内存。
  • 堆很容易发生内存泄漏,其中内存被分配给未使用的对象,并且除此之外的进程将无法使用。
  • 堆中没有大小限制。
  • 与堆栈相比,堆中的对象访问速度要慢得多。写入堆上的内存也比较慢。

堆栈使用起来更容易,更快,但它带来了很多限制,如果你使用堆,你可以忽略它们。

你什么时候使用堆栈? 堆栈只能用于占用少量内存的局部变量。好消息是内存分配和管理不会成为您的问题,访问这些对象的速度非常快。它确实受到大小限制以及无法在堆栈上调整变量的事实。

你什么时候使用堆?如果存在需要全局访问的变量,则使用堆来分配内存,而不是只对创建它的方法和函数可用。当你需要大量内存时,堆也很好,因为它对内存大小没有限制。您还可以在堆上调整变量的大小。

以上是关于JVM中的堆和非堆内存的主要内容,如果未能解决你的问题,请参考以下文章

jdk的内存设置

tomcat编译内存溢出怎么解决

Java常见问题分析

深入了解JVW

Java堆内存Heap与非堆内存Non-Heap

JVM内存溢出及配置