8年开发,一直不知道 Java为什么要加 final 关键字!.docx

8年开发,一直不知道 Java为什么要加 final 关键字!.docx

  1. 1、本文档共7页,可阅读全部内容。
  2. 2、原创力文档(book118)网站文档一经付费(服务费),不意味着购买了该文档的版权,仅供个人/单位学习、研究之用,不得用于商业用途,未经授权,严禁复制、发行、汇编、翻译或者网络传播等,侵权必究。
  3. 3、本站所有内容均由合作方或网友上传,本站不对文档的完整性、权威性及其观点立场正确性做任何保证或承诺!文档内容仅供研究参考,付费前请自行鉴别。如您付费,意味着您自己接受本站规则且自行承担风险,本站不退款、不进行额外附加服务;查看《如何避免下载的几个坑》。如果您已付费下载过本站文档,您可以点击 这里二次下载
  4. 4、如文档侵犯商业秘密、侵犯著作权、侵犯人身权等,请点击“版权申诉”(推荐),也可以打举报电话:400-050-0827(电话支持时间:9:00-18:30)。
查看更多
8年开发,一直不知道 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();//---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 =

文档评论(0)

150****6040 + 关注
实名认证
内容提供者

互联网产品运营推广以及k12教育内容。

1亿VIP精品文档

相关文档