终于明白 Java 为什么要加 final 关键字了!

沙海 2021年6月16日12:11:19Java评论32字数 2759阅读9分11秒阅读模式
摘要

终于明白 Java 为什么要加 final 关键字了! 程序员闪充宝

终于明白 Java 为什么要加 final 关键字了!

程序员闪充宝 文章源自JAVA秀-https://www.javaxiu.com/32636.html

收录于话题文章源自JAVA秀-https://www.javaxiu.com/32636.html

#Java4文章源自JAVA秀-https://www.javaxiu.com/32636.html

#程序员闪充宝1文章源自JAVA秀-https://www.javaxiu.com/32636.html

终于明白 Java 为什么要加 final 关键字了!

来源:https://www.jianshu.com/p/acc8d9a67d0c文章源自JAVA秀-https://www.javaxiu.com/32636.html

  • 现象描述文章源自JAVA秀-https://www.javaxiu.com/32636.html

  • 原因分析文章源自JAVA秀-https://www.javaxiu.com/32636.html

  • 情景对比文章源自JAVA秀-https://www.javaxiu.com/32636.html

在开发过程中,由于习惯的原因,我们可能对某种编程语言的一些特性习以为常,特别是只用一种语言作为日常开发的情况。文章源自JAVA秀-https://www.javaxiu.com/32636.html

但是当你使用超过一种语言进行开发的时候就会发现,虽然都是高级语言,但是它们之间很多特性都是不太相同的。文章源自JAVA秀-https://www.javaxiu.com/32636.html

现象描述

在 Java 8 之前,匿名内部类在使用外部成员的时候,会报错并提示 “Cannot refer to a non-final variable arg inside an inner class defined in a different method”文章源自JAVA秀-https://www.javaxiu.com/32636.html

终于明白 Java 为什么要加 final 关键字了!文章源自JAVA秀-https://www.javaxiu.com/32636.html

below-java8.jpg文章源自JAVA秀-https://www.javaxiu.com/32636.html

但是在 Java 8 之后,类似场景却没有再提示了:文章源自JAVA秀-https://www.javaxiu.com/32636.html

终于明白 Java 为什么要加 final 关键字了!文章源自JAVA秀-https://www.javaxiu.com/32636.html

normal-use.jpg文章源自JAVA秀-https://www.javaxiu.com/32636.html

难道是此类变量可以随便改动了吗?当然不是,当你试图修改这些变量的时候,仍然会提示错误:文章源自JAVA秀-https://www.javaxiu.com/32636.html

终于明白 Java 为什么要加 final 关键字了!文章源自JAVA秀-https://www.javaxiu.com/32636.html

try-to-change.jpg文章源自JAVA秀-https://www.javaxiu.com/32636.html

可以看到,当试图修改基本数据类型的变量时,编译器的警告变成了 “Varible 'num' is accessed from within inner class, need to be final or effectively final”文章源自JAVA秀-https://www.javaxiu.com/32636.html

很遗憾,仍然不能修改。相比之下,Kotlin 是没有这个限制的:文章源自JAVA秀-https://www.javaxiu.com/32636.html

终于明白 Java 为什么要加 final 关键字了!文章源自JAVA秀-https://www.javaxiu.com/32636.html

usage-in-kt.jpg文章源自JAVA秀-https://www.javaxiu.com/32636.html

原因分析

从表面上当然看不出什么原因,看看编译器做了什么工作吧!运行 javac 命令后生成了几个 .class 文件:文章源自JAVA秀-https://www.javaxiu.com/32636.html

终于明白 Java 为什么要加 final 关键字了!文章源自JAVA秀-https://www.javaxiu.com/32636.html

generated-files.jpg文章源自JAVA秀-https://www.javaxiu.com/32636.html

不难推断,这个 TestInnerClass$1.class 就是匿名内部类编译后的文件,看看它反编译后是什么内容:文章源自JAVA秀-https://www.javaxiu.com/32636.html

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);    }}

原来,匿名内部类也会被当作普通的类处理,只不过编译器生成它构造方法的时候,除了将外部类的引用传递了过来,还将基本数据类型的变量复制了一份过来,并把引用数据类型的变量引用也传递了过来。文章源自JAVA秀-https://www.javaxiu.com/32636.html

因此,基本数据类型的变量当然不能修改了,不然就会跟外部的变量产生不一致,这样的话变量的传递也就变得毫无意义了。文章源自JAVA秀-https://www.javaxiu.com/32636.html

final 关键字除了能让类不能被继承之外,对应到这种场景,就是让变量也不能被重新赋值。文章源自JAVA秀-https://www.javaxiu.com/32636.html

情景对比

但是为什么对于 Kotlin 来说可以在匿名内部类中直接修改基本数据类型的值呢?查看 Kotlin 编译后反编译回来的内容:文章源自JAVA秀-https://www.javaxiu.com/32636.html

   publicfinal void useNestedClass(@NotNullfinal 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);   }

可以发现,当需要传递基本数据类型的变量时,Kotlin 编译器会将这些数据进行包装,从而由值传递变为引用传递,这样内部的修改当然就不会影响到外部了。文章源自JAVA秀-https://www.javaxiu.com/32636.html

验证一下,当变量不进行传递时,Kotlin 编译器是怎么处理的:文章源自JAVA秀-https://www.javaxiu.com/32636.html

   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);   }
一款基于 Spring Boot 的BBS系统,APP和后台管理齐全,拿来即用(附项目地址)大厂为什么都很重视 API 网关?聊聊 API 网关的作用有了HTTP,为什么还要RPC?使用 JWT 来保护你的 SpringBoot 应用快手二面:a==1 && a==2 && a==3 是 true 还是 false?Spring Boot + Spring Security + Thymeleaf 实战教程!SpringBoot注解最全详解(整合超详细版本)为什么不推荐使用BeanUtils属性转换工具!老程序员都不用...Docker + FastDFS + Spring Boot 一键式搭建分布式文件服务器八幅漫画理解使用 JWT 设计的单点登录系统
文章源自JAVA秀-https://www.javaxiu.com/32636.html

阅读原文文章源自JAVA秀-https://www.javaxiu.com/32636.html

继续阅读
速蛙云 - 极致体验,强烈推荐!!!购买套餐就免费送各大视频网站会员!快速稳定、独家福利社、流媒体稳定解锁!速度快,全球上网、视频、游戏加速、独立IP均支持!基础套餐性价比很高!这里不多说,我一直正在使用,推荐购买:https://www.javaxiu.com/59919.html
weinxin
资源分享QQ群
本站是JAVA秀团队的技术分享社区, 会经常分享资源和教程; 分享的时代, 请别再沉默!
沙海
匿名

发表评论

匿名网友 填写信息

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定