丑话说在前
学习笔记
第二章
第一节: 内存管理##
-
运行时内存
- 方法区
- 虚拟机栈
- 本地方法栈
- 堆
- 程序计数器
-
程序计数器:当前线程所执行的字节码的行号指示器。
-
Java虚拟机多线程:线程轮流切换并分配处理器执行时间。
-
虚拟机栈:每个方法被执行的时候会创建一个栈帧用于储存局部变量表,操作栈,动态链表,方法出口等信息。每一个方法调用直至执行完成的过程,就对应一个栈帧在虚拟机中入栈到出栈的过程。
-
虚拟机栈的局部变量表部分:在编译期完成分配
- 1.编译器可知的基本数据类型:boolean、byte、char、shot、int、float、long、double
- 2.对象引用
- 3.返回地址
-
虚拟机规范中对局部变量表规定两种异常状况
- 1.如果线程请求的栈深度大于虚拟机允许的深度,抛出stackOverflowError
- 2.当虚拟机栈扩展无法申请到足够的内存时会抛出OutOfMemoryError
-
虚拟机栈与本地方法栈
- 虚拟机栈为虚拟机执行java方法(字节码)服务
- 本地方法栈为虚拟机使用到的Native方法服务
-
java堆
- 1.堆由所有线程共享
- 2.用于存放对象实例以及数组
- 3.是垃圾收集器管理的主要区域,称为GC堆,垃圾堆hh
- 4.如果堆中没有内存完成实例分配,并且堆也无法再拓展时,将抛出OutOfMemoryError异常
-
方法区
- 1.由所有线程共享
- 储存已被加载的类信息、常量、静态变量、编译后的代码数据
-
方法区-运行时常量池
- Class文件中除了有类的版本、字段、方法、接口等描述信息外,还有常量池。
- 具备动态性,没有特殊要求
- 编译器和运行期间都可以将常量放入运行时常量池
-
直接内存
- 直接内存并不属于JVM的内存结构,它是物理机的内存,但是JVM虚拟机可以调用该部分内存。
-
对象访问:
-
例 Object obj = new Object();
-
1.Object obj 会反映到java栈的本地变量表中,作为一个reference类型数据。
-
2.new Object() 会反映到java堆中,形成一块储存Obejct类型所有实例数据值的结构化内存
-
3.java堆中还包含能查到该对象类型数据的地址信息
-
4.类型数据储存在方法区中
-
5.实现方式: 句柄、直接指针
- 句柄方式:在java堆中划分出一块内存做句柄池,reference储存的就是对象的句柄的地址。句柄中包含了对象实例数据和类型数据各自的具体地址信息。优势:reference中储存的是稳定的句柄地址,在对象被移动,如垃圾回收,时只会改变句柄中的实例数据指针。
- 直接指针:reference中直接储存对象地址。优点:访问速度块。
-
-
虚拟机参数
- -Xms10m,-Xmx20m:堆最小空间10M,堆最大空间20M
- -Xss10m:本地方法栈大小10m
- -XX:PermSize=10M, -XX:MaxPermSize=10M: 限制方法区的大小,1.8后已经弃用,并且将永久代并入堆,使用-XX:MaxMetaspaceSize=128m
- -XX:MaxDirectMemorySize=10M,设置直接内存最大值
第三章
垃圾处理与内存分配
-
判断对象“已死”
- 1.引用计数算法:
- 给对象加一个引用计数器,有一个地方引用,就加一,引用失效,就减一。为0就是不再使用。优点:实现简单,判定效率高。缺点:难以解决对象之间的循环引用问题。
- 2.根搜索算法:
- 通过一系列名为GC Root的对象作为起始点,从这些节点开始向下搜索,搜索所走过的路径称为引用链,当一个对象到GC Roots没有任何引用链相连,则对象不可用
- GC Roots对象包括:虚拟机栈中的引用对象、方法区中的类静态属性引用的对象、方法区中常量引用的对象、Native方法的引用的对象
- 1.引用计数算法:
-
四种引用类型(jdk1.2之后):
- 1.强引用
- 类似Object obj = new Object()
- 只要强引用还存在,垃圾收集器就不会回收引用的对象。
- 2.软引用
- 用来描述一些还有用,但不是必需的对象。
- 在系统将要发生
- 1.强引用