5.5 清理:终结处理和垃圾回收
Java中使用垃圾回收器负责回收无用对象占据的内存资源。但是有特殊情况,假定你的对象(并非使用new创建)获得了一块特殊的内存区域,由于垃圾回收器只知道回收那些经由new分配的内存,所以它不知道该如何释放该对象的这块特殊内存。为了应对这种情况,Java允许类定义一个名为finalize()的方法,一旦垃圾回收器准备好释放该对象占用的存储空间,将首先调用其finalize()方法,并在下一次垃圾回收动作真正发生时才会真正回收对象占用的内存。所以定义finalize()方法,能在垃圾回收时刻做一些重要的清理工作。
之所以要有finalize(),是由于在分配内存时可能使用了类似C语言中的做法,例如在非Java代码中调用了C的malloc()函数系列来分配存储空间,而且除非调用了free()函数否则存储空间将无法得到释放,从而造成内存泄漏。当然,free()是C语言中的函数,所以需要在finalize()中使用本地方法调用。
如果希望进行处了释放存储空间之外的清理工作,还是得明确调用某个恰当的方法(而绝不能直接调用finalize方法)。因为无论是垃圾回收还是终结方法,都不保证一定会发生。如果Java虚拟机并未面临内存耗尽的情形,它是不会浪费时间去执行垃圾回收已恢复内存的。
finalize()还有一个用法,那就是对象终结条件的验证。如果某个对象在被回收时必须处于某种特定的状态,那么可以在终结方法中进行验证。
垃圾回收器如何工作(P90-91):(1)垃圾回收器对于提高对象的创建速度,具有明显的效果。
(2)Java虚拟机的堆实现更像一个传送带,每分配一个新对象,它就往前移动一格。这意味着对象存储空间的分配速度非常快。Java的堆指针只是简单的移动到尚未分配的区域,其效率了比得上C++在堆栈上的分配空间的效率。
(3)垃圾回收器工作时,将一面回收空间,一面使堆中的对象紧凑排列,这样堆指针就可以很容易的移动到更靠近传送带的开始处,也就尽量避免了页面错误。通过垃圾回收器对对象重排列,实现了一种高速的,有无限空间可分配的堆模型。
(4)Java虚拟机中有许多附加技术用意提升速度,尤其是与加载器操作有关的,被称为(Just-In-Time,JIT)编译器的技术。这种技术把程序全部或部分翻译成本地机器码(将字节码编译成本地机器码),程序运行速度因此得到提升。
5.7 初始化
(1)可以使用构造器进行初始化,但是无法阻止自动初始化的进行,它将在构造器被调用之前发生。
public class Counter{
int i;
public Counter(){
i=7;
}
}
那么i首先会被置为0,然后变成7。
(2)在类的内部,成员变量会在任何方法(包括构造器)被调用之前得到初始化。
总结一下对象的创建过程,假设有一个名为Dog的类:
(1)构造器实际上是静态方法,当首次创建Dog对象或者Dog类的静态方法/静态域被首次访问时,Java解释器查找类路径,以定位Dog.class文件。
(2)载入Dog.class(创建一个Class对象),执行静态初始化的所有动作。因此,静态初始化只在Class对象首次加载的时候进行一次。
(3)当用new Dog() 创建对象是,首先在堆上分配存储空间,这块存储空间会被清零(成员变量设为默认值)。
(4)执行所有字段定义处的初始化动作。
(5)执行构造器。
5.9 枚举类型
当你创建enum时,编译器会自动创建toString方法,static values()方法,返回按照声明顺序构成的enum数组,ordinal()方法返回一个枚举值在enum中的声明位置。
for (Color c:Color.values()){
System.out.println(c+",ordinal:"+c.ordinal());
}