Android 安卓技术分享

【专题】Java技术体系

2018-06-06  本文已影响180人  都有米

这是《深入理解Java虚拟机》的读书笔记。

java生态

1、Java技术体系

Java技术体系

2、Java运行时数据区

成员变量
* 成员变量定义在类中,在整个类中都可以被访问。
* 成员变量随着对象的建立而建立,随着对象的消失而消失,存在于对象所在的堆内存中。
* 成员变量有默认初始化值。
局部变量
* 局部变量只定义在局部范围内,如:函数内,语句内等,只在所属的区域有效。
* 局部变量存在于栈内存中,作用的范围结束,变量空间会自动释放。
* 局部变量没有默认初始化值

Java运行时数据区

程序计数器、虚拟机栈、本地方法栈三个区域随线程生而生,虽线程灭而灭。其中栈帧随着方法的进入和退出而有条不紊的执行着入栈和出栈操作。每个栈帧分配多少内存在类结构确定下来时就已知了,这几个区域内存分配和回收都具有稳定性,不需要过多考虑回收问题,线程灭就自动回收了。java堆和方法区的内存都是动态分配和回收的。方法区进行垃圾收集“性价比”比较低,主要还是管理Java堆。

对象访问

主流的访问方式有两种:句柄和直接指针。在各种语言和框架中这两种访问方式都很常见,各有千秋。

句柄访问
java堆中划出一块内存作为句柄池,句柄中包含对象实例数据和类型数据各自的具体地址信息。最大的好处就是reference中存储的是稳定的句柄地址,在对象被移动的时候只改变句柄的实例数据指针。(垃圾收集时移动对象是非常常见的现象)

句柄访问

直接指针
Java堆中的对象布局必须考虑如何防止类型数据地址。reference中存储的直接就是对象地址。最大的好处就是速度快,节省一次指针定位的时间开销。

直接指针

再谈引用

引用:当reference类型的数据中存储的数值代表的是另一块内存的起始地址,就称其为引用。引用有如下4中类型:

3、垃圾收集器

确定对象是否已死

在Java中可作为GC Roots的对象有
* 虚拟机栈中引用的对象(栈帧中本地变量表)
* 方法区中的类静态属性引用的对象
* 方法区中的常量引用的对象
* 本地方法栈帧中JNI引用的对象

收集器判断对象不可达后,会检查对象的finalize()方法是否被执行过,如果没有就先执行finalize()。一个对象的finalize()只会被执行一次,如果一个对象在finalize()方法中被救活了,即重新被引用了,当它再次不可达时不会执行finalize()方法了。

垃圾收集算法

优点:这是最基础的收集算法,其他的算法都是它的基础上演变来的。
缺点:1、效率问题,标记和清除效率都不高;2、空间问题,标记清除后产生大量的不连续的内存碎片。

优点:实现简单、运行效率高。
缺点:可使用的内存缩小为原来的一半
改进:IBM研究表面,新生代中98%的对象都是朝生夕死,所以并不需要1:1来划分。所以分成一块较大的Eden空间和a、b两块较小的Survivor空间(Eden:Survivor = 8:1)。每次使用Eden和一块Survivor空间,比如Survivor-a,用完了就把存活的对象拷到Survivor-b上,然后清除Eden和Survivor-a,然后再使用Eden和Survivor-b。每次浪费10%的空间,如果存活对象大于10%就把多出来的对象放到给新生代担保的老年代。

目前商业虚拟机都采用改进后的复制算法回收新生代。

4、类加载

Class文件需要被加载到虚拟机中才能被运行和使用。生命周期包括:加载(Loading)、验证(Verification)、准备(Preparation)、解析(Resolution)、初始化(Initialization)、使用(Using)和卸载(Unloading)七个阶段。

类加载器

虚拟机中类的唯一性由类本身和它的加载器决定。

双亲委派模型的父子关系不是继承,而是组合关系。


双亲委派模型

双亲委派模型的工作工程:类加载器收到类加载请求时,首先不会尝试自己加载这个类,而是把这个请求委派给父类加载器去完成,每一层加载器都是如此,所以所有的加载请求都会传递到启动类加载器,只有父加载器反馈自己无法完成这个加载请求时(它的搜索范围中没有找到所需的类),子加载器才会尝试自己去加载。

上一篇下一篇

猜你喜欢

热点阅读