一、Java内存区域理解
Java与C++之间有一堵由内存动态分配(加载)和垃圾收集技术所围成的高墙,墙外面的1人想进去,墙内边的人想出去。
1.Jvm内存:
- Jvm在执行java程序的过程中会把Jvm所管理的内存划分为若干个不同数据区域。 方法区 Method Area、虚拟机栈VM Stack、本地方法栈Native Method Stack、堆Heap、程序计数器Program Counter Register。
- 注:方法区、堆运行时所有数据是线程共享的。栈、本地栈、程序计数器运行时所有数据是线程独享的。
- Person:存放在元空间,也可以说方法区;
- person:存放在Java栈的局部变量表中;
- new Person():存放在Java堆中。
2.方法区:1.7永久代(堆)、1.8元空间(本地内存) https://www.jianshu.com/p/87354570f362
方法区的定义:方法区是Jvm规范中定义的规范,主要用于存储已被虚拟机加载的【Class】类型信息、常量、静态变量、即时编译器编译后的代码缓存等。其大小可通过参数调节。
方法区的垃圾回收:方法区的垃圾收集主要回收两部分内容:常量池中废弃的常量和不再使用的类型。
类型信息:对每个加载的类型(类class、接口interface、枚举enum、注解annotation),JVM必须在方法区中存储以下类型信息:
1)这个类型的完整有效名称(全名=包名.类名)。
2)这个类型直接父类的完整有效名(对于interface或是java.lang.object,都没有父类)。
3)这个类型的修饰符(public,abstract,final的某个子集)。
4)这个类型直接接口的一个有序列表。
域信息:JVM必须在方法区中保存类型的所有域的相关信息以及域的声明顺序:
1)域的相关信息包括:域名称、域类型、域修饰符(public,private,protected,static,final,volatile,transient的某个子集)。
方法信息:JVM必须保存所有方法的以下信息,同域信息一样包括声明顺序:
1)方法名称
2)方法的返回类型(或void)
3)方法参数的数量和类型(按顺序)
4)方法的修饰符(public,private,protected,static,final,synchronized,native,abstract的一个子集)
5)方法的字节码(bytecodes)、操作数栈、局部变量表及大小(abstract和native方法除外)
6)异常表(abstract和native方法除外),每个异常处理的开始位置、结束位置、代码处理在程序计数器中的偏移地址、被捕获的异常类的常量池索引
静态类变量:静态变量和类关联在一起,随着类的加载而加载,他们成为类数据在逻辑上的一部分。
1)静态类变量被类的所有实例共享,即使没有类实例时,你也可以访问它。
方法区的实现:JDK1.7及以前:Hotspot虚拟机对方法区的实现是永久代,方法区和永久代的关系就类比如Java中的接口和实现类。永久代就是HotSpot虚拟机对Jvm虚拟机规范中方法区的一种实现方式。此时的方法区是堆的逻辑组成部分。《Java虚拟机规范》中明确说明:“尽管所有的方法区在逻辑上是属于堆的一部分,但一些简单的实现可能不会选择去进行垃圾收集或者进行压缩。”但对于 HotSpotJVM 而言,方法区还有一个别名叫做Non-Heap(非堆),目的就是要和堆分开。所以,方法区看作是一块独立于Java堆的内存空间。
JDK1.7方法区(永久代)的特点:
1)方法区(Method Area)与Java堆一样,是各个线程共享的内存区域。
2)方法区在JVM启动的时候被创建,并且它的实际的物理内存空间中和Java堆区一样都可以是不连续的。
3)方法区的大小,跟堆空间一样,可以选择固定大小或者可扩展。
4)方法区的大小决定了系统可以保存多少个类,如果系统定义了太多的类,导致方法区溢出,虚拟机同样会抛出内存溢出错误:java.lang.OutofMemoryError:PermGen space 或者java.lang.OutOfMemoryError:Metaspace。
5)关闭JVM就会释放这个区域的内存。
方法区的实现:JDK1.8:Hotspot虚拟机对方法区的实现是