前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >一分钟告诉你java final 关键字运行原理

一分钟告诉你java final 关键字运行原理

作者头像
ImportSource
发布2018-04-03 16:45:47
7240
发布2018-04-03 16:45:47
举报
文章被收录于专栏:ImportSourceImportSource

final关键字究竟是怎么运行的?

这是一个非常有趣的问题,

import java.util.ArrayList; import java.util.List; class Test { private final List foo; public Test() { foo = new ArrayList(); foo.add("foo"); // Modification-1 } public void setFoo(List foo) { // this.foo = foo; //compile time error. } }

在上面的demo代码中,我们定义了一个Test构造函数,并且定义了一个setFoo方法。

关于构造函数:构造函数只在实例化一个对象的时候才被调用一次,通过new这个关键字。我们不能在创建一个对象的时候调用多次构造函数,因为构造函数就是这样规定的。

关于方法:方法可以被调用很多次。编译器也知道我们有可能调用这个方法很多次。所以这会就会抛出编译错误,编译自然无法通过。

第一种情况:

private final List foo; // 1

foo是一个实例变量(instance variable)。当我们创建Test类的对象以后,实例变量foo就会被

为Test类的对象。如果我们在构造函数内给foo赋值,那么编译器知道构造函数只能被调用一次,所以编译器不会抛错出来。所以我们在构造函数里给foo赋值没有什么问题。

如果我在方法里边给foo赋值的话,也可以叫实例化,那么编译器知道方法可能要被调用多次。这时候foo的值要被修改多次,但final变量并不允许这样做。我们只能对final变量赋值一次,only one time。所以让构造函数来给final变量赋值是一个不错的选择,good choice!

第二种情况:

private static final List foo = new ArrayList();

现在foo是一个静态的变量。所以当我们创建一个Test类的对象,foo变量就不会被复制到Test类的对象中,因为foo现在是静态的。现在的foo不再是每个对象的属性。现在它是Test类的属性。但是foo对所有的对象都是可见的,如果使用new关键字来创建对象,那么每次都要调用构造函数,这样的话,foo的值就会在多个对象创建过程中被修改,所以也是有问题的。final是不能被修改的。

第三种情况:

t.foo.add("bar"); // Modification-2

上面这种情况就是被允许的,因为你不是去修改第一次赋值的那个对象引用,而是在foo里边增加内容,这并没有改变对象引用。编译器只有在你给foo赋值一个新的ArrayList的时候才会不通过,才会抱怨。总之,规则就是如果已经给final的引用变量初始化了对象,赋了值,那么你就不能再去修改它,不能再把一个新的对象赋值给它。比如这里的ArrayList。

这里顺便说下final的其他。总之

final类不能有子类,这个相信大家都知道。

final方法不能再被重写。

final 方法可以重写。注意被重写和重写是两码子事。就是你可以重写一个方法,然后把它在此阉割变为final的意思。

本文参与 腾讯云自媒体分享计划,分享自微信公众号。
原始发表:2016-05-03,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 ImportSource 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档