java final关键字_终于明白 Java 为什么要加 final 关键字了!
在開發過程中,由于習慣的原因,我們可能對某種編程語言的一些特性習以為常,特別是只用一種語言作為日常開發的情況。但是當你使用超過一種語言進行開發的時候就會發現,雖然都是高級語言,但是它們之間很多特性都是不太相同的。
現象描述
在 Java 8 之前,匿名內部類在使用外部成員的時候,會報錯并提示 “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
可以看到,當試圖修改基本數據類型的變量時,編譯器的警告變成了 “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 就是匿名內部類編譯后的文件,看看它反編譯后是什么內容:
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);} }原來,匿名內部類也會被當作普通的類處理,只不過編譯器生成它構造方法的時候,除了將外部類的引用傳遞了過來,還將基本數據類型的變量復制了一份過來,并把引用數據類型的變量引用也傳遞了過來。因此,基本數據類型的變量當然不能修改了,不然就會跟外部的變量產生不一致,這樣的話變量的傳遞也就變得毫無意義了。final 關鍵字除了能讓類不能被繼承之外,對應到這種場景,就是讓變量也不能被重新賦值。
情景對比
但是為什么對于 Kotlin 來說可以在匿名內部類中直接修改基本數據類型的值呢?查看 Kotlin 編譯后反編譯回來的內容:
public final void useNestedClass(@NotNull final TestNestedClass.DataBean bean) {Intrinsics.checkParameterIsNotNull(bean, "bean");final IntRef num = new IntRef();//---1num.element = 1;//---2String var3 = "before action, num = " + num.element;System.out.println(var3);<undefinedtype> nestedClass = new TestNestedClass.NestedClass() {public void doSomething() {num.element = 678;//---3bean.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;//---4System.out.println(var4);}可以發現,當需要傳遞基本數據類型的變量時,Kotlin 編譯器會將這些數據進行包裝,從而由值傳遞變為引用傳遞,這樣內部的修改當然就不會影響到外部了。驗證一下,當變量不進行傳遞時,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);}哈哈,并沒有多此一舉,點個贊!
有想跟小編一起學習的可以跟我交流!
《新程序員》:云原生和全面數字化實踐50位技術專家共同創作,文字、視頻、音頻交互閱讀總結
以上是生活随笔為你收集整理的java final关键字_终于明白 Java 为什么要加 final 关键字了!的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: python社区发现_这个 Python
- 下一篇: php flock 都是true_php