“不可变的对象”与“不可变的对象引用”
什么是不可变的对象呢?我们都知道String是不可变的,如果有涉及大量的字符串拼接我们最好不要用String,虽然我们在代码中可以这样写:
String str = "test"; str = "test1";
这样写是没有错的,这是不是和我们所说的String类型时不可变的违背了呢?其实不然,我们这里看到str“改变”了值,其实不是不是真正改变了,而是改变了str的引用。我们从下图可以看到当定义String str = "test1"时,Java实际上做了这个操作(我们在这里不讨论String关于使用new和不使用new,以及堆内存分配的问题)。
在我们再一次赋值str = "test1"时,Java实际上做了下面这个操作:
"test"变量其实并没有改变,改变的只是str的引用,将str的引用重新指向在常量池中新创建的"test1"变量,这即是"不可变的对象"。
那么何为"不可变的对象引用"呢?字面意思当然就是和上面相反,允许它改变str的引用重新指向"test1"字符串变量。即是在定义变量时加上final关键字。我们看以下代码:首先有一个User类,这个类中的name变量是普通变量,可以通过setName方法赋值。
1 package day_20_immutable; 2 3 /** 4 * @author 余林丰 5 * 6 * 2016年10月20日 7 */ 8 public class User { 9 private String name; 10 11 public String getName() { 12 return name; 13 } 14 15 public void setName(String name) { 16 this.name = name; 17 } 18 }
我们在main方法中来试验一把final关键字的“不可变的对象引用”。
1 package day_20_immutable; 2 3 /** 4 * @author 余林丰 5 * 6 * 2016年10月20日 7 */ 8 public class Main { 9 private final User user = new User(); 10 /** 11 * @param args 12 */ 13 public static void main(String[] args) { 14 Main main = new Main(); 15 main.user.setName("keven"); 16 System.out.println(main.user.getName()); 17 main.user.setName("turbo"); //final关键字不影响本身是可改变的对象 18 System.out.println(main.user.getName()); 19 20 //main.user = new User(); //报错,因为final关键字修饰的变量不能改变它的引用 21 22 } 23 24 }
以上便是“不可变对象”与“不可变的对象引用”的区别,为什么要区分这两个概念,这是为后面Java多线程的线程安全先做下铺垫。