final、finally和finalize的区别是什么?
三者的区别如下:
1)final属于声明属性、方法和类,分别表示属性不可变,方法不可覆盖,类不可被继承(不能再派生出新的子类)。
final属性:被final修饰的变量不可变,由于不可变有两重含义,一是引用不可变,二是对象不可变。那么final到底指的是哪种含义呢?下面通过一个例子进行说明:
public class Test{
public static void main(String[]args){
final StringBuffer s=new StringBuffer("Hello");
s.append("world");
System.out.println(s);
}
}
运行结果为:Hello world
public class Test{
public static void main(String[]args){
final StringBuffer s=new StringBuffer("Hello");
s=new StringBuffer("Hello world");
}
}
编译期间错误
从上述例子可以看出,final指向的是引用的不可变性,即它只能指向初始时指向的那个对象,而不关心指向对象内容的变化。所以,被final修饰的变量,必须被初始化。一般可以通过以下几种方式对其进行初始化:1.在定义的时候初始化;2.final成员变量可以在初始化块中初始化,但不可在静态初始化块中初始化;3.静态final成员变量可以在静态初始化块中初始化;4.在类的构造器中初始化,但静态final成员变量不可以在构造方法中初始化;
final方法:当一个方法声明为final时,该方法不允许任何子类重写这个方法,但子类仍然可以使用这个方法。另外还有一种被称为inline(内联)的机制,当调用一个被声明为final的方法时,直接将方法主体插入到调用处,而不是进行方法调用(类似于C++语言中的inline),这样做能提高程序的效率;
final参数:用来表示这个参数在这个方法的内部不允许被修改;
final类:当一个类别声明为final时,此类不能被继承,所有的方法不能被重写。但这并不表示,final类的成员变量也是不可更改的,要想做到final类的成员变量不可改变,必须给成员变量增加final修饰。值得注意的是,一个类既不能被声明为abstract,又被声明为final。
2)finally作为异常处理的一部分,它只能在try/catch语句中,并且附带着一个语句块,表示这段语句最终一定被执行,经常被用在需要释放资源的情况下。示例1:不使用finally的代码如下所示:
Connection conn;
Statement stmt;
try{
String url="jdbc:sqlserver://localhost:1433;DatabaseName=dbName";
conn=DriverManager.getConnection(url,"username","password");
stmt=conn.createStatement();
stmt.executeUpdate(update);
stmt.close();
conn.close();
}catch(Exception){
}
在上面的程序片段中,如果程序在运行过程中没有发生异常,那么数据库的连接能够得到释放,程序运行没有问题。如果在执行update语句时出现异常,后面的close()方法将不会被调用,数据库的连接将得不到释放。如果有大量这种程序的运行,可能会耗光数据库的连接资源。通过使用finally可以保证任何情况下数据库的连接资源都能够被释放。示例2:使用finally代码如下所示:
Connection conn=null;
Statement stmt=null;
try{
String url="jdbc:sqlserver://localhost:1433;DatabaseName=dbName";
conn=DriverManager.getConnection(url,"username","password");
stmt=conn.createStatment();
stmt.executeUpdate("select *from table");
stmt.close();
conn.close();
}catch(Exception e){
exception handling;
}finally{
if(stmt!=null){
try{
stmt.close();
}catch(SQLException e){
}
}
if(conn!=null){
try{
stmt.close();
}catch(SQLException e){
}
}
}
在上面的代码中,不管程序运行是否会出现异常,finally中的代码一定会执行,这样能保证在任何情况下,数据库的连接都能被释放。
3)finalize是Object类的一个方法,在垃圾收集器执行的时候会调用被回收对象的finalize()方法,可以覆盖此方法来实现对其他资源的回收,例如关闭文件等。需要注意的是,一旦垃圾回收器准备好释放对象占用的空间,将首先调用其finalize()方法,并且在下一次垃圾回收动作发生时,才会真正回收对象占用的内存。