Java中关键字final finally finalize的区别介绍(Java中关键字有哪些)

网友投稿 309 2022-08-09


Java中关键字final finally finalize的区别介绍(Java中关键字有哪些)

目录1. final1.1 final修饰属性1.2 final修饰方法1.3 final修饰类2. finally3. finalize

这三个除了长得像以外,好像没什么联系

1. final

final意为“最后的”,它是java中的一个关键字。

final可以修饰属性、方法、类。

1.1 final修饰属性

从final的含义就不难理解用final修饰内容的用意。final修饰属性,就表示这个属性是“最终的”,也就是不可更改的,换成我们熟悉的名词,也就是“常量”。

private final double PI = 3.1415926; //被final修饰的属性常常用大写表示,全部大写在idea的快捷键是ctrl+shift+u

此时如果再想修改这个值,就会报错。

final修饰的属性可以通过多种方式进行初始化,比如显式初始化、代码块初始化、构造方法初始化等等。

public class Test {

private final int WIDTH = 10; //显示初始化

private final int HEIGHT;

private final int LEFT;

{

HEIGHT = 10; //代码块初始化

}

public Test(){

LEFT = 10; //构造方法初始化

}

public Test(int n){

LEFT = n; //构造方法初始化

}

}

final修饰属性经常和static一起使用,表示全局常量 。

另外,final还可以修饰局部变量

public void test(final int NUMBER){

}

public void test(int number){

final int NUM = number;

}

上文提到final修饰属性的多种初始化方法,这里注意一点,切记不要使用普通方法去初始化,当然这种方式都不会编译通过。因为final引用的属性在对象出现前就已经存在了,调用方法赋值就太晚了。

1.2 final修饰方法

final修饰方法表示此方法不可重写。套用含义上的理解,也就是“最终的方法”,也就是不可被修改的方法。

比如写一个父类

class ParentClass{

public final void finalMethod(){

}

}

尝试在子类中重写这个方法,就会报错。

1.3 final修饰类

final修饰类,套用含义上的理解,“最终的类&rdqhttp://uo;意为不可修改的类,表示该类不可被继承,因为继承就可以重写方法、扩展功能,是对现有类的修改。

声明一个final的类

final class ParentClass{

}

尝试去继承这个类,就会报错

2. finally

finally用于异常处理,try-catch-finally,finally表示最终会执行的功能块。finally包裹的代码块一定会执行。(注意这个“一定会执行”,后文有坑)

try{

//可能出现异常的代码

}

catch(异常类型1 变量名1){

//处理异常的方式1

}

catch(异常类型2 变量名2){

//处理异常的方式2

}

finally{

//一定会执行的代码

}

举一个例子

@Test

public void test(){

try{

int a = 10;

int b = 0;

System.out.println(a/b);

}

catch (ArithmeticEhttp://xception e){

System.out.println("处理方式1");

}

catch (Exception e){

System.out.println("处理方式2");

}

finally {

System.out.println("执行finally");

}

}

执行结果:

注意以下三点:

finally是可选的,在try-catch结构中不使用finally没有任何问题finally声明的代码是一定会执行的像数据库连接、输入输出流、网络Socket… JVM是不能自动回收的,所以常常在可能存在异常的代码中手动在finally释放掉这些资源

扩展-finally和return谁会先执行?

上文多次提到“finally声明的代码是一定会执行的”,那么在try或者catch中出现了return,那么谁会先执行?

执行以下代码,会输出什么?

private int finallyTest(){

try{

int a = 10;

int b = 0;

System.out.println(a/b);

return 1;

}

catch (Exception e){

return 2;

}

finally {

System.out.println("执行finally");

}

}

调用一下

@Test

public void test(){

int num = finallyTest();

System.out.println(num);

}

结果如下图,也就是即使在catch中有return语句,也要等待finally声明的代码执行完再return出去。

3. finalize

finalize翻译为“使结束”,它是一个方法,是每个类默认存在的方法。

打开Object类的源码,就可以找到finalize()方法

finalize()方法用于GC回收,finalize是本文最难理解的概念,要真正理解finalize()方法,就要深入理解GC回收机制,而本文的侧重点并不在于解释GC回收机制,所以只会简单地描述一下finalize()方法的使用场景,读者如果感兴趣,请自行查找相关的文章。

在Java的内存管理中,使用可达性分析算法来判断对象是否存活,基本思路是通过一系列称为“GC Roots”的跟对象作为起始节点,根据引用关系向下搜索,如果某些对象到GC Roots之间没有任何引用关系,则证明此对象是不能被再使用的,就有可能去回收这一块内存了。下图中,虽然object5、object6、object7之间有引用关系,但是到GC Roots没有任何途径,则这些对象仍在回收的范围内。

生存还是死亡?

即使经过可达性分析算法判定为不可达的对象,也不是非要回收不可,要真正回收一个内存空间,至少需要两次的不可达判定。经过第一次不可达判定,随后要进行一个筛选,筛选的条件是该对象是否有必要执行finalize()方法。假如待回收的对象没有重写finalize()方法,或者finalize()方法被JVM调用过(每个对象的finalize()方法只能被JVM调用一次),这两种情况都会被虚拟机认为是“没有必要执行finalize()方法”,那么在经过几轮不可达标记后,该对象被真正地回收掉。

如果该对象被认定为“有必要执行finalize()方法”,则稍后会被低调度优先级的线程去执行finalize()方法,而重写的finalize()方法有可能完成一次对象的“自救”,比如将this赋给某个属性,那么在后续标记时会被判定为“可达”,那么JVM就不会回收这个对象。

尽管finalize()方法有着特殊的使用场景,但是永远也不要显式调用某个对象的finalize()方法,应该交给GC回收机制调用。


版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。

上一篇:springboot中使用redis并且执行调试lua脚本(redis lua脚本实现事务)
下一篇:Java实现登录和注册案例(javaweb注册登录案例)
相关文章

 发表评论

暂时没有评论,来抢沙发吧~