final:
Java语言关键字,用于声明属性、方法和类。分别表示属性不可变,方法不可覆盖,类不可继承。将变量或方法声明为final,可以保证它们在使用中不被改变。
1、被声明为final的变量必须在声明时给定初值,而在以后的引用中只能读取,不可修改。
2、被声明为final的方法也同样只能使用,不能被重载。
3、被声明为final的类表示此类为最终类,即它不能再派生出新的子类,不能作为父类被继承。因此一个类不能既被声明为 abstract的,又被声明为final的。
finally:
异常处理语句结构的一部分,表示总是执行。通常可在异常处理时提供 finally 块来执行任何清除操作。如果抛出一个异常,那么相匹配的 catch 子句就会执行,然后控制就会进入 finally 块(如果有的话)。
finalize:
Object 类的一个方法,因此所有的类都继承了它。Java 技术允许使用 finalize() 方法在垃圾收集器将对象从内存中清除出去之前做必要的清理工作。这个方法是由垃圾收集器在确定这个对象没有被引用时,在删除对象之前调用的,因此子类可以 覆盖 finalize() 方法以整理系统资源或者执行其他清理工作。
如下为一个综合示例:
* 此类为最终类,不可以被继承
* @author Amon
*/
public final class Test
{
/**
* 此属性是只读的
*/
private final String sChar = "";
/**
* 默认构造器
*/
public Test()
{
}
/**
* 类方法,不可以被修改
*/
public final static void staticMethod()
{
}
/**
* 实例方法,不可以被重写
*/
public final void instanceMethod()
{
}
/**
* 重写了Object类的析构方法,用于善后处理
*/
@Override
protected void finalize()
{
}
}
【延伸阅读一】
finalize 是位于Object类的一个方法,该方法的访问修饰符为protected,由于所有类为Object的子类,因此用户类很容易访问到这个方法。由 于,finalize函数没有自动实现链式调用,我们必须手动的实现,因此finalize函数的最后一个语句通常是super.finalize()。 通过这种方式,我们可以实现从下到上实现finalize的调用,即先释放自己的资源,然后再释放父类的资源。
根据Java语言规范,JVM保证调用finalize函数之前,这个对象是不可达的,但是JVM不保证这个函数一定会被调用。另外,规范还保证finalize函数最多运行一次。
很 多Java初学者会认为这个方法类似与C++中的析构函数,将很多对象、资源的释放都放在这一函数里面。其实,这不是一种很好的方式。原因有三,其 一,GC为了能够支持finalize函数,要对覆盖这个函数的对象作很多附加的工作。其二,在finalize运行完成之后,该对象可能变成可达 的,GC还要再检查一次该对象是否是可达的。因此,使用finalize会降低GC的运行性能。其三,由于GC调用finalize的时间是不确定的,因 此通过这种方式释放资源也是不确定的。
通常,finalize用于一些不容易控制、并且非常重要资源的释放,例如一些I/O的操 作,数据的连接。这些资源的释放对整个应用程序是非常关键的。在这种情况下,程序员应该以通过程序本身管理(包括释放)这些资源为主,以finalize 函数释放资源方式为辅,形成一种双保险的管理机制,而不应该仅仅依靠finalize来释放资源。
【延伸阅读二】
当垃圾 回收器(garbage colector)决定回收某对象时,就会运行该对象的finalize()方法。值得C++程序员注意的是,finalize()方法并不能等同与析构 函数。Java中是没有析构函数的。C++的析构函数是在对象消亡时运行的。由于C++没有垃圾回收,对象空间手动回收,所以一旦对象用不到时,程序员就 应当把它delete()掉。所以析构函数中经常做一些文件保存之类的收尾工作。但是在Java中很不幸,如果内存总是充足的,那么垃圾回收可能永远不会 进行,也就是说filalize()可能永远不被执行,显然指望它做收尾工作是靠不住的。
那么finalize()究竟是做什么 的呢?它最主要的用途是回收特殊渠道申请的内存。Java程序有垃圾回收器,所以一般情况下内存问题不用程序员操心。但有一种JNI(Java Native Interface)调用non-Java程序(C或C++),finalize()的工作就是回收这部分的内存。