<kbd id="afajh"><form id="afajh"></form></kbd>
<strong id="afajh"><dl id="afajh"></dl></strong>
    <del id="afajh"><form id="afajh"></form></del>
        1. <th id="afajh"><progress id="afajh"></progress></th>
          <b id="afajh"><abbr id="afajh"></abbr></b>
          <th id="afajh"><progress id="afajh"></progress></th>

          終于明白 Java 為什么要加 final 關鍵字了!

          共 4347字,需瀏覽 9分鐘

           ·

          2021-06-18 08:06

          來源:https://www.jianshu.com/p/acc8d9a67d0c

          • 現(xiàn)象描述

          • 原因分析

          • 情景對比


          在開發(fā)過程中,由于習慣的原因,我們可能對某種編程語言的一些特性習以為常,特別是只用一種語言作為日常開發(fā)的情況。

          但是當你使用超過一種語言進行開發(fā)的時候就會發(fā)現(xiàn),雖然都是高級語言,但是它們之間很多特性都是不太相同的。

          現(xiàn)象描述

          在 Java 8 之前,匿名內(nèi)部類在使用外部成員的時候,會報錯并提示 “Cannot refer to a non-final variable arg inside an inner class defined in a different method”

          below-java8.jpg

          但是在 Java 8 之后,類似場景卻沒有再提示了:

          normal-use.jpg

          難道是此類變量可以隨便改動了嗎?當然不是,當你試圖修改這些變量的時候,仍然會提示錯誤:

          try-to-change.jpg

          可以看到,當試圖修改基本數(shù)據(jù)類型的變量時,編譯器的警告變成了 “Varible 'num' is accessed from within inner class, need to be final or effectively final”

          很遺憾,仍然不能修改。相比之下,Kotlin 是沒有這個限制的:

          usage-in-kt.jpg

          原因分析

          從表面上當然看不出什么原因,看看編譯器做了什么工作吧!運行 javac 命令后生成了幾個 .class 文件:

          generated-files.jpg

          不難推斷,這個 TestInnerClass$1.class 就是匿名內(nèi)部類編譯后的文件,看看它反編譯后是什么內(nèi)容:

          class TestInnerClass$1 extends InnerClass {
              TestInnerClass$1(TestInnerClass var1, int var2, DataBean var3) {
                  super(var1);
                  this.this$0 = var1;
                  this.val$num = var2;
                  this.val$bean = var3;
              }

              void doSomething() {
                  super.doSomething();
                  System.out.println("num = " + this.val$num);
                  System.out.println("bean name is: " + this.val$bean.name);
              }
          }

          原來,匿名內(nèi)部類也會被當作普通的類處理,只不過編譯器生成它構造方法的時候,除了將外部類的引用傳遞了過來,還將基本數(shù)據(jù)類型的變量復制了一份過來,并把引用數(shù)據(jù)類型的變量引用也傳遞了過來。

          因此,基本數(shù)據(jù)類型的變量當然不能修改了,不然就會跟外部的變量產(chǎn)生不一致,這樣的話變量的傳遞也就變得毫無意義了。

          final 關鍵字除了能讓類不能被繼承之外,對應到這種場景,就是讓變量也不能被重新賦值。

          情景對比

          但是為什么對于 Kotlin 來說可以在匿名內(nèi)部類中直接修改基本數(shù)據(jù)類型的值呢?查看 Kotlin 編譯后反編譯回來的內(nèi)容:

             public final void useNestedClass(@NotNull final TestNestedClass.DataBean bean) {
               Intrinsics.checkParameterIsNotNull(bean, "bean");
               final IntRef num = new IntRef();//---1
               num.element = 1;//---2
               String var3 = "before action, num = " + num.element;
               System.out.println(var3);
               <undefinedtype> nestedClass = new TestNestedClass.NestedClass() {
                  public void doSomething() {
                     num.element = 678;//---3
                     bean.setName("xyz");
                     String var1 = "num = " + num.element;
                     System.out.println(var1);
                     var1 = "bean name is: " + bean.getName();
                     System.out.println(var1);
                  }
               };
               nestedClass.doSomething();
               String var4 = "after action, num = " + num.element;//---4
               System.out.println(var4);
            }

          可以發(fā)現(xiàn),當需要傳遞基本數(shù)據(jù)類型的變量時,Kotlin 編譯器會將這些數(shù)據(jù)進行包裝,從而由值傳遞變?yōu)橐脗鬟f,這樣內(nèi)部的修改當然就不會影響到外部了。

          驗證一下,當變量不進行傳遞時,Kotlin 編譯器是怎么處理的:

             public final void useNestedClass(@NotNull TestNestedClass.DataBean bean) {
                Intrinsics.checkParameterIsNotNull(bean, "bean");
                int num = 1;
                String var3 = "before action, num = " + num;
                System.out.println(var3);
                int num = 678;
                var3 = "after action, num = " + num;
                System.out.println(var3);
             }

          瀏覽 59
          點贊
          評論
          收藏
          分享

          手機掃一掃分享

          分享
          舉報
          評論
          圖片
          表情
          推薦
          點贊
          評論
          收藏
          分享

          手機掃一掃分享

          分享
          舉報
          <kbd id="afajh"><form id="afajh"></form></kbd>
          <strong id="afajh"><dl id="afajh"></dl></strong>
            <del id="afajh"><form id="afajh"></form></del>
                1. <th id="afajh"><progress id="afajh"></progress></th>
                  <b id="afajh"><abbr id="afajh"></abbr></b>
                  <th id="afajh"><progress id="afajh"></progress></th>
                  最新大香蕉视频久久网 | 亚州操逼视频 | 疯狂做爰XXXⅩa久久久久久 | 成人在线视频黄色 | 伊人大香蕉视频 |