你真的知道final关键字吗?

释放双眼,带上耳机,听听看~!

概述

final在英文中是最终的,不可更改的。在Java中final修饰变量,函数和类,就像这个单词的意思,一旦使用赋值之后不可更改。

final修饰的变量不可以被改变

finalTest类

public class finalTest { private String test = \"test\"; //final修饰的成员变量 第一种赋值方式 private final int a = 6; private final int b; private final int c; //final修饰的静态成员变量 第一种赋值方式 private final static int e = 6; private final static int f; public finalTest() { //final修饰的成员变量 第二种赋值方式 b = 5; } { //final修饰的成员变量 第三种赋值方式 c = 5; } static { //final修饰的静态成员变量 第二种赋值方式 f = 6; } public void test(final int g) { //final 修饰的局部变量第一种方式 final int d = 10; //final 修饰的局部变量第二种赋值方式 final int h; h = 11; //调用此函数的已经进行赋值,再次赋值会报错 //g = 66;  } @Override public String toString() { return test; } public static void main(String[] args) { final finalTest finalTest = new finalTest(); //final修饰,无法被改变 //finalTest = null;  System.out.println(finalTest); finalTest.test = \"test2\"; System.out.println(finalTest); } }

输出

test
test2

final修饰基本数据类型,不能对基本数据类型重新赋值,基本数据类型变量不能被改变。但final修饰引用类型变量,不变的仅仅是他的一个引用,只要引用地址不变,他里面的成员变量是可变的。

final修饰的函数不可以被重写

finalTest类

public class finalTest { public final void test() { System.out.println(\"父类\"); } } class finalTestChild extends finalTest { // @Override // public void test() // { //
// } //final修饰的函数可以被重载
    public void test(int a) { System.out.println(\"重载\"); } }

 

父类被final修饰的函数是无法被子类重写的,但final修饰的函数可以被重载。

final修饰的类不可以被继承

最典型是的例子是Java的String类,打开String类发现String类是被final所修饰。

finalTest类

你真的知道final关键字吗?

继承一个被final修饰的父类,就会报错。很明显Java设计人员不希望我们对String类进行修改,个人理解是因为String类过于强大,Java的设计人员出于安全考虑,不希望它有子类。 因为这样可能危及到系统安全。所以final类中所有的类都隐式指定为是final的,无法覆盖他们,我们只能使用他规定的函数。所以当我们希望自己的类不被人继承时,就可以指定为final。

final的作用

  1. 效率,JVM和Java都会缓存final变量并对函数,和类进行优化。
  2. 设计和安全,上\"锁\",不希望自己的类和函数被人随意改变。

final的安全发布

创建一个对象,大致可以分为三个步骤

  1. 分配内存空间
  2. 将引用指向分配的内存空间
  3. 调用构造函数来初始化对象

这三个步骤不是原子的,执行到第二部,没进行初始化,此时如果这个对象能被当前范围之外的代码所使用,因为这时对象已经不是null了,被其他代码访问,会得到一个错误的结果。这就是不安全的发布。所谓安全发布,简单理解就是对象的创建能够保障在被别人使用前,完成数据的构造设置,或者说一个对象使用时,已经进行初始化。但是Java对此并没有进行保障,需要自己进行保障设置,如锁机制等。

对于final,当创建一个对象时,使用final关键字能够使得另一个线程不会访问到处于\"部分创建\"的对象。

  1. 当构造函数退出时,final字段的值保证对访问构造对象的其他线程是可见的。
  2. 一旦对象引用对其他线程可见,则其final成员也必须正确赋值。

所以借助final,就如同是你对对象的创建访问加锁了一样,天然保障了对象的安全发布

总结

许多并发错误都是没理解共享对象的既定规则,当发布一个对象时,必须明确说明对象的访问方式。

 

给TA打赏
共{{data.count}}人
人已打赏
随笔日记

[jvm]运行时数据区域详解

2020-11-9 4:04:32

随笔日记

目标检测——IoU 计算

2020-11-9 4:04:34

0 条回复 A文章作者 M管理员
    暂无讨论,说说你的看法吧
个人中心
购物车
优惠劵
今日签到
有新私信 私信列表
搜索