三、浅谈final、finally、finalize有什么不同

一、final可以用来修饰类、方法、变量。final修饰的类表示不能继承扩展,final的变量是不可以修改的,而final的方法也是不可以重写的。

  1. 我们可以将方法或者类声明为final,这样可以明确告知别人,这些行为是不能修改的。

  2. 使用final修饰参数或者变量,也可以清楚地避免意外赋值导致的编程错误,甚至,有人明确推荐将所有方法参数,本地变量,成员变量声明成final。

  3. final在Java内存模型中的语义规则:
    (1)在一个线程中,初次读对象引用与初次读该对象包含的final域,JVM禁止处理器重排序这两个操作(这个规则仅仅针对处理器)。编译器会在读final域之前插入一个LoadLoad屏障。
    (2)JVM禁止编译器把final域的写重排序到构造函数之外。写final字段的重排序规则可以确保,在对象引用为任意线程可见之前,对象的final字段已经正确初始化过了,而普通字段不具有这个保障。

  4. final修饰的变量在某种程度上产生了不可变的效果,所以可以用于保护只读数据,尤其是在并发编程中,因为明确地不能再赋值final变量,有利于减少额外的同步开销,也可以省去一些防御性拷贝的必要。
    但是final并不等于不可变,比如下面这段代码:

      final List<String> strList = new ArrayList<>();
      strList.add("Hello");
      strList.add("world");
      List<String> unmodifiableStrList = List.of("hello", "world");
      unmodifiableStrList.add("again"); 

final只能约束strList这个引用不可以不赋值,但是strList对象行为不被final影响,添加元素等操作完全是正常的。如果我们希望对象本身是不可变的,那么需要相应的类支持支持不可变的行为。
实现一个不可变类的步骤有:
(1)将class声明为final,这样别人就不能通过扩展来绕开限制了。
(2)将所有成员变量定义为private和final,并且不要实现setter方法
(3)通常构造对象时,成员变量使用深度拷贝来初始化,而不是直接赋值,这是一种防御措施,因为你无法确定输入对象不被其他人修改。
(4)如果确实需要实现getter方法,或者其他可能返回内部状态的方法,使用copy-on-write原则,创建私有的copy。

  1. final也许会有性能好处,很多文章或者书籍都介绍了特定场景提高性能,比如,利用final可能有助于JVM将方法内联,可以改善编译器进行条件编译的能力等等。坦白说,很多类似的结论都是基于假设得出的,
    比如现代高性能JVM(如HotSpot)判断内联未必依赖final的提示。final字段对性能的影响,大部分情况下并没有考虑的必要。在日常开发中,除非有特别的考虑,不然最好不要指望这种小技巧带来的所谓性能好处。

二、finally是保证Java重点代码一定要执行的一种机制。我们可以使用try-finally或者try-catch-finally来进行类似JDBC连接、保证解锁等动作。需要注意下面这种代码会输出什么内容。

   try {
         // do something
         System.exit(1);
     } finally{
         System.out.println("Print from finally");
     }

三、finalize是基础类java.lang.object的一个方法,它的设计目的是保证对象在垃圾收集前完成特定资源的回收。finalize机制现在已经不推荐使用,并且在JDK9中被标记为Deprecated。
finalize存在的问题:
(1)finalize的执行是和垃圾收集关联在一起的,一旦实现了非空的finalize方法,就会导致对象回收呈现数量级的变慢。因为finalize被设计成对象被垃圾回收之前调用,这就意味着实现了finalize方法的对象是个
"特殊公民",JVM要对他进行额外的处理。finalize本质上成为了快速回收的阻碍者,可能导致对象会经过多个垃圾收集周期才能回收。
(2)finalize还会掩盖资源回收时的出错信息,java.lang.ref.Finalizer类的源码如下所示,这就意味着一旦出现异常或者错误,应用得不到任何有效的错误信息。

    private void runFinalizer(JavaLangAccess jla) {
        synchronized (this) {
            if (hasBeenFinalized()) return;
            remove();
        }
        try {
            Object finalizee = this.get();
            if (finalizee != null && !(finalizee instanceof java.lang.Enum)) {
                jla.invokeFinalize(finalizee);

                /* Clear stack slot containing this variable, to decrease
                   the chances of false retention with a conservative GC */
                finalizee = null;
            }
        } catch (Throwable x) { }
        super.clear();
    }


(3)java.lang.Object#finalize方法注释写到,finalize方法引起的任何异常都会导致此对象的回收被停止,这可能会导致内存泄漏的问题。

     * Any exception thrown by the {@code finalize} method causes
     * the finalization of this object to be halted, but is otherwise
     * ignored.
     *
     * @throws Throwable the {@code Exception} raised by this method
     * @see java.lang.ref.WeakReference
     * @see java.lang.ref.PhantomReference
     * @jls 12.6 Finalization of Class Instances
     */
    protected void finalize() throws Throwable { }


(4)有什么机制可以替换finalize吗?
Java平台目前正在逐步使用java.lang.ref.Cleaner来替换掉原有的finalize实现。Cleaner的实现利用了幻想引用,这是一种常见的所谓post-mortem清理机制,利用幻想引用和引用队列,我们可以保证对象被彻底晓辉之前做一些类似于资源回收的工作,比如关闭文件描述符等操作系统有限的资源,它比finalize更加轻量、更加可靠。Cleaner的操作都是独立的,它有自己的运行线程,所以可以避免意外死锁等问题。下面是JDK提供的样例程序:

public class CleaningExample implements AutoCloseable {
        // A cleaner, preferably one shared within a library
        private static final Cleaner cleaner = <cleaner>;
        static class State implements Runnable { 
            State(...) {
                // initialize State needed for cleaning action
            }
            public void run() {
                // cleanup action accessing State, executed at most once
            }
        }
        private final State;
        private final Cleaner.Cleanable cleanable
        public CleaningExample() {
            this.state = new State(...);
            this.cleanable = cleaner.register(this, state);
        }
        public void close() {
            cleanable.clean();
        }
    }

从可预测性的角度来判断,Cleaner或者幻想引用改善的程度仍然是有限的,如果由于种种原因导致幻想引用堆积,同样会出现问题。所以,Cleaner适合作为最后一种保证手段,而不是完全依赖Cleaner进行资源回收。资源用完就显式释放才是比较安全的。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,928评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,192评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,468评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,186评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,295评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,374评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,403评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,186评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,610评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,906评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,075评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,755评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,393评论 3 320
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,079评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,313评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,934评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,963评论 2 351

推荐阅读更多精彩内容