java中final和finally和finalize的区别

final、finally、finalize傻傻分不清楚,今天让你彻底弄清楚

基础概念区分

final

修饰符

  1. 将类声明为final,意味着它不能再派生新的子类,不能作为父类被继承。final类中的所有成员方法都会被隐式地指定为final方法。

(因此一个类不能及被声明为abstract,又被声明为final的。)
2. 将方法声明为final,只能使用,不能被子类方法重写。
3. 将变量声明为final,必须在声明时给定初值。如果是基本数据类型的变量,则其数值一旦在初始化之后便不能更改;如果是引用类型的变量,则在对其初始化之后便不能再让其指向另一个对象。

finally

异常处理语句结构的一部分,表示总是执行。

try-catch-finally(可省略)

finalize

Object类的一个方法

  1. 在垃圾回收器执行的时候会调用被回收对象的此方法,可以覆盖此方法提供垃圾收集时的其他资源回收,例如关闭文件等。
  2. 该方法更像是一个对象生命周期的临终方法,当该方法被系统调用则代表该对象即将“死亡”,但是需要注意的是,我们主动行为上去调用该方法并不会导致该对象“死亡”,这是一个被动的方法(其实就是回调方法),不需要我们调用。
  3. 在GC要回收某个对象时,这个对象:“最后一刻,我还能再抢救一下!”。因此JVM要对它进行额外处理。finalize成为了CG回收的阻碍者,导致这个对象经过多个垃圾收集周期才能被回收。
public class Person {
    private String name;
    private int age;

    public Person(String name, int age) {
        this.name = name;
        this.age = age;
    }

    public String toString() {
        return "姓名:" + this.name + ",年龄:" + this.age;
    }

    public void finalize() throws Throwable {//对象释放空间是默认调用此方法
        System.out.println("对象被释放-->" + this);//直接输出次对象,调用toString()方法
    }

}

class SystemDemo {

    public static void main(String[] args) {
        // TODO Auto-generated method stub
        Person per = new Person("zhangsan", 30);
        per = null;//断开引用,释放空间
        //方法1:
        System.gc();//强制性释放空间
        //方法2:
//        Runtime run=Runtime.getRuntime();
//        run.gc();
    }
}

try、catch、finally执行顺序

try、catch、finally为异常捕获结构,其中finally为非必需项,可有可无。当finally存在时,finally块中的代码都会执行。

try-catch结构;
try-catch-finally结构;
try-finally结构也是可以的。

特殊场景:
仅在以下四种情况下不会执行finally块中语句。

  1. 如果在try或catch语句中执行了System.exit(0);
  2. 在执行finally之前jvm崩溃了
  3. try语句中执行死循环
  4. 电源断电

不管有无异常,finally中代码都会执行

public class FinallyTest {
    // 当传入参数str 为 "null" 时抛出异常
    public static void throwTest(String str) {
        if ("null".equals(str)) {
            throw new NullPointerException();
        }
    }

    public static String method(String str) {
        try {
            System.out.println("try");
            throwTest(str);
        } catch (Exception e) {
            System.out.println("exception: " + e);
        } finally {
            System.out.println("finally");
        }
        return str;
    }

    public static void main(String[] args) {
        System.out.println("-----无异常-----");
        method("nl");
        System.out.println("-----有异常-----");
        method("null");
    }
}

上述代码做了正常执行和抛出异常的测试,结果如下:

-----无异常-----
try
finally
-----有异常-----
try
exception: java.lang.NullPointerException
finally

try中有return时,finally依然会执行

public static String method(String str) {
    try {
        System.out.println("try--");
        return str;
    } catch (Exception e) {
        System.out.println("exception-- " + e);
    } finally {
        System.out.println("finally--");
    }
    return "end";
}

public static void main(String[] args) {
    System.out.println(method("str"));
}

上述代码做了try中返回测试,结果如下:

try--
finally--
str

finally对返回值的做修改,不会影响到try的返回值

public static String method(String str) {
    try {
        System.out.println("try--");
        return str;
    } catch (Exception e) {
        System.out.println("exception-- " + e);
    } finally {
        System.out.println("finally--");
        str = "finally";
    }
    return "end";
}

public static void main(String[] args) {
    System.out.println(method("str"));
}

上述代码做了在finally中修改返回值的测试,最终返回依然为"str",结果如下:

try--
finally--
str

finally是在return后面的表达式运算后执行的(此时并没有返回运算后的值,而是先把要返回的值保存起来,不管finally中的代码怎么样,返回的值都不会改变,任然是之前保存的值),所以函数返回值是在finally执行前确定的。

finally中包含return语句,程序会在finally中提前退出

public static String method(String str) {
    try {
        System.out.println("try--");
        return str;
    } catch (Exception e) {
        System.out.println("exception-- " + e);
    } finally {
        System.out.println("finally--");
        str = "finally";

        // 在finally中的return 语句,会造成程序提前退出
        return str;
    }
}

public static void main(String[] args) {
    System.out.println(method("str"));
}

上述代码做了在finally中,使用return语句的测试,程序会在finally中提前退出,结果如下:

try--
finally--
finally

经典测试题

面题目输出什么?

public static int demo5() {
    try {
        return printX();
    }
    finally {
        System.out.println("finally trumps return... sort of");
    }
}

public static int printX() {
    System.out.println("X");
    return 0;
}

输出结果:

X
finally trumps return... sort of
0

上面这道题目含金量很高,程序顺序执行时先执行printX()函数,此时得到返回值0并且将0保存到variable中对应的用于保存返回值的区域;

此时程序在执行finally语句因为finally语句中没有return语句,所以程序将返回值区域的0返回给上一级函数。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

栋幺栋幺-

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值