final:可以修饰类,方法,变量;
- 对类的修饰:表示类不可以被继承;
- 对方法修饰:表示该方法不能被子类重写;
- 对变量修饰:表示该变量不能被修改;
不知道大家有没有遇到一些很奇怪的现象,就是被final修饰变量好像能被修改;
比如下面这段代码吧
public class Model {
public String id;
public String userName;
public String password;
public Model() {
id ="model_id";
userName="model_userName";
password = "model_password";
}
public class Test {
public static final Model model = new Model();
public static void main(String[] args) {
//修改前
System.out.println(model.toString());
//修改id的值;
model.setId("fix_model_id");
//修改后
System.out.println(model.toString());
}
这个model对象被final修饰了,正常来讲不应该修改才对啊!那为什么被final修饰之后还会被修改呢?
要理解这个问题,还必须要了解一点Java虚拟机的知识,首先看下Java虚拟机运行时数据区:
虚拟机栈(VM Stack)描述的是Java方法执行的内存模型,每个方法在执行的时候都会创建一个叫栈帧(Stack Frame)的东西用来存储**局部变量表**,操作数栈,动态链接,方法出口等信息。
这其中很重要的信息虚拟机栈中会创建:局部变量表。
局部表量表中有什么?
局部变量表:各种基本数据类型(int,short,long,float,double,char,boolean,byte),引用对象(reference类型,他不同于对象本身,可能是是一个指向对象起始地址的指针;也可能是指向一个对象的句柄或与此对象相关的位置);
也就是说,如果变量不是基本类型,那么存储的值我们可以理解为指向实际对象的内存地址,如下图;
上面的例子:
public static final Model model = new Model();
mode变量对应的值,是一个引用地址;这个地址对应的值是真正的model对象;
因而,变量model被final修饰之后,如果引用的model对象没有变,那么就是正确;但是修改model对象的属性值是可以的,因为final只是确保引用的model对象不变就行;
打个比方:你有张三的电话号码(对象的引用),张三胖了30斤(对象属性值变化),但是你持有的引用仍然还是张三;final的作用相当于就是让你对张三的引用不要变,能随时引用到张三;不管他变胖,变瘦,变高,变矮,他只要是张三就行;但是如果从张三的引用,变成了对张三双胞胎弟弟的引用就不行了会报错;因为持有的引用已经变了,尽管属性值一模一样但是本质变了;
下图就是一个错误引用:
- 原引用: model -> 0xffff ;
- 新引用: model -> 0x5fff;
下图更改引用对象,就会导致报错;
回到最开始的问题,被final修饰的变量的值为什么还是会被修改?
其实看到这里,问题的答案已经很明显了。被final修饰的变量值是没有变的;
public static final Model model = new Model();
被final修饰的model,存储的只是一个内存地址,setId只是通过引用地址找到正真的model实例数据,修改了model对象的实例数据;而model存储的值(指向对象的指针)并没有变,疑惑解除~;