<返回更多

Java基础系列-final、finally关键字

2021-12-15    唯一浩哥
加入收藏

一、概述

final是JAVA关键字中最常见之一,表示“最终的,不可更改”之意,在Java中也正是这个意思。

有final修饰的内容,就会变得与众不同,它们会变成终极存在,其内容成为固定的存在。

finally关键字不同于final关键字,这是一个需要与异常体系结构配合使用的关键字,旨在定义必须要进行操作,一般用于在发生异常的时候进行一些收尾操作,比如释放资源等。

另外还有个finalized,它是一个方法,它需要与垃圾收集体系配合使用。主要在对象被垃圾收集之前进行一些操作,这些操作只会被执行一次,即使一个对象多次被标记为下次进行垃圾收集,也只有第一次会执行。

二、final作用

2.1 final修饰变量

变量被final修饰就会变成为常量,常量被保存在方法区中。

变量一旦被final修饰,必须手动进行初始化,未进行初始化的final常量是无法通过编译的。

如果只有final修饰的变量的初始化可以采用:

如果被static和final同时修饰的变量的初始化可以采用:

一旦final变量被static修饰,那么它就脱离了对象的组织(代码块、构造器都是对象的组织),升级为类的组织,所以需要在类级别的静态代码块中进行初始化。

public class FinalTest {
    final int i = 1;
    int j = 2;
    static int m = 3;
    static final int n = 4;
}

public class FinalTest {
    final int i;
    int j;
    static int m;
    static final int n;
    {
        i = 1;
    }
    static {
        n = 3;
    }
}

如果将上面的代码改成:

public class FinalTest {
    final int i;//2-
    int j;
    static int m;
    static final int n;//5-
}

上面代码第2行和第5行会报错,原因就是未进行初始化。

那么我们总结下final和static的现象,用于区分二者:

注意:同时被final和static修饰的变量成为静态常量,类常量,这种类常量在编译阶段会进行常量传播优化,将该类常量的值直接保存到调用类的常量池中,那么在程序执行到调用位置时,实际上与定义该类常量的类已经毫无关系,我(调用方)可以直接在我的常量池中获取到编译阶段优化过来的值,不再需要通过常量定义类的类型去调用其中定义的那个类常量了。

二者可以同时存在,各起各的作用。

2.2 final修饰方法

被final修饰的方法,可以被子类继承,但是不能被子类重写,也就是说这个方法在此以后其内部的实现就是固定不变的了,不能被改变。

2.3 final修饰类

被final修饰的类,被称之为最终类,其不再拥有子类,不可再进行扩展,最常见的final类就是String类。

String类被final修饰之后,其每个对象都是不变的,一旦定义就不再发生改变。

2.4 final修饰局部变量

final修饰的局部变量,该变量就不再是保存在栈空间中,而是保存在方法区中,不会随方法结束而失效,放大了局部变量的生命周期。

最常使用的地方就是局部内部类在访问方法的局部变量的情况下,这些局部变量就需要使用final修饰,因为当局部内部类访问局部变量时,会放大局部变量的作用域,局部变量一般在方法结束时就失效了,但是却有可能任然被内部类的对象持有使用。将该局部变量定义为final之后,它不再保存于栈空间,而是保存在方法区中,自然不会因为方法的结束而丢失。

public class FinalTest{
    public void outMethod(){
       final int s = 1;// 3-
        class innerClass{
            public void innerMethod(){
                System.out.println(s);
            }
        }
    }
}

如果去掉第3行的final,第5行就会报错。

2.5 final修饰方法参数

如果方法的参数被final修饰,那么这个参数的值在从方法调用时赋值开始就不能再改变,不能被重新赋值(不能改成他值)。

public class FinalTest{
    public void outMethod(final int s){
       s = 1;
    }
}

如上代码,方法参数s为final的,那么若去掉第2行的代码,为s重新赋值,则会报错。

三、finally作用

finally只有一种用法,那就是在try...catch..语句末尾使用。语法如下:

public class FinallyTest{
    public void test(){
        try{
          //someExecute
        }catch(Exception e){
          //someExecute
        }finally{
          //someExecute
        }
    }
}

finally块中的语句是一定会被执行的,无论是否会发生异常,重点:这里的异常指的是在try块中的部分,如果实在try块之前发生了异常,还没来得及执行try块语句,那么finally块中的内容也不会被执行,所以finally针对的是try块中的内容而设的。

如果在try块或catch块中存在return语句,那么,finally块中的内容必然会在return之前执行。

finally经常用于发生异常的情况下关闭打开的资源,比如io流,网络资源等。

四、finalized作用

finalized是Object类的protected方法。

当垃圾回收器发现一个对象不存在任何引用的时候,就会触发该方法的调用,调用由垃圾回收器发起。

子类重写该方法一般用于处理系统资源或者一些清理工作。

该方法并不被确保一定会调用,但是可以保证的是,一旦被调用,调用的线程并不会持有任何的同步锁,而且如果执行发生了异常,则忽略异常,同时停止执行。也就是说,finalized方法并不会对程序的正常流程、代码的正常运行造成意外的影响。

class FinalizedTest{
    @Override
    public void finalize(){
        // do something
    }
}

简述finalized执行流程:

当对象变成(GC Roots)不可达时,GC会判断该对象是否覆盖了finalize方法,若未覆盖,则直接将其回收。否则,若对象未执行过finalize方法,将其放入F-Queue队列,由一低优先级线程执行该队列中对象的finalize方法。执行finalize方法完毕后,GC会再次判断该对象是否可达,若不可达,则进行回收,否则,对象“复活”。(摘自参考文章)

参考:

声明:本站部分内容来自互联网,如有版权侵犯或其他问题请与我们联系,我们将立即删除或处理。
▍相关推荐
更多资讯 >>>