每个区域的作用:
- 方法区(线程共享)
存放类的二进制字节码文件(二进制字节码文件中包括:类基本信息,常量池,类方法定义,包含了虚拟机指令);
所有java虚拟机共享的区,存储了和类的结构、类信息
类结构:运行时常量池(含字符串常量),静态变量,类的信息,常量
类信息:魔数,版本号,常量池,类(字段和方法),父类和接口数组,字段,方法等信息。
可以结合类加载器理解:所有上述的信息在Class对象中都可以获取到
方法区逻辑上是堆的一部分,不同厂商在具体实现时不一定,方法区的位置不固定,方法区会产生内存溢出;
方法区是JVM中的一个规范,永久代和元空间是方法区的两个不同实现;
Jdk1.8之前使用永生代实现,1.8之后使用元空间实现。
-
堆(线程共享)
通过new关键字创建的对象都会由堆存储;堆中不再被引用的对象会通过垃圾回收机制自动回收
细分一点的内存结构:年轻代、老年代;垃圾回收策略也不同。
逻辑上又分为:新生代(Eden、from、to)、老年代
- jvm栈(线程私有)
线程运行时需要的内存空间;
每个线程只能有一个活动栈帧,对应着一个线程正在执行的那个方法,该栈帧处于栈顶。
一个栈内是由多个栈帧组成的,一个栈帧对应着一个方法的调用,每个栈帧都有自己独立的存储空间
问题辨析:
垃圾回收是否涉及栈内存?
不涉及,栈帧在每次方法执行完毕会自动排出栈,即被回收。
栈内存分配越大越好吗?
可通过-Xss 改变栈内存分配大小;栈内存分配大了更多的是能够执行的方法更多了,那么相对的能够分配给线程的栈数量就减少了。
方法内的局部变量是否线程安全?
其实就是看变量是共享还是私有的
如果方法内局部变量没有逃离方法的作用访问,即私有的,则是线程安全;
如果局部变量引用了对象,并逃离了方法的作用,需要考虑线程安全
- 程序计数器(线程私有)
为了线程切换后能恢复到正确的执行位置,每个线程有一个程序计数器,物理上程序计数器是通过寄存器(CPU组件中读取速度最快的单元)来实现的,程序计数器是线程私有的,是java内存中唯一不会存在内存溢出的区
- 本地栈(线程私有)
本地方法栈的功能特点类似于虚拟机栈,每个线程单独创建自己的本地方法区,主要用于保存本地方法信息。
不同处:虚拟机栈服务的是java方法,本地方法栈服务的是Native方法。
本地方法举例:hashCode,notify,wait
常见参数
堆:‐Xms2048M ‐Xmx2048M
堆-新生代:‐Xmn1024M
栈:‐Xss512K
方法区:‐XX:MetaspaceSize=256M ‐XX:MaxMetaspaceSize=256M
为什么最大与最小值设置成一样的?
参考链接:
JVM内存结构_好久啦的博客-优快云博客_jvm内存结构JVM学习之-JVM内存结构 - 大漠驼铃 - BlogJava
jvm调优技巧 - 内存抖动 、Xms和Xmx参数为什么要设置相同的值_java叶新东老师的博客-优快云博客_xms和xmx为什么要相同