首页 > 代码库 > “不可变的对象”与“不可变的对象引用”

“不可变的对象”与“不可变的对象引用”

什么是不可变的对象呢?我们都知道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多线程的线程安全先做下铺垫。

 

“不可变的对象”与“不可变的对象引用”