java中mutable(可变)和immutable(不可变)的关系与使用
-
各自含义
不变数据类型(immutable type):指一旦被创建,其值不能改变,只能用于访问。immutable类型必定没有任何修改对象状态的方法,即没有mutator这种方法类型(例如setter)。在java中经常使用的数据类都是immutable的,如 Integer, Long,String等。
可变类型(mutable type):拥有方法可以修改自己的值,即必然存在mutator方法类型,能改变对象状态或数据表示。例如List的add方法就能够修改数据表示。
本质上来所,不可变数据类型更改后地址发生改变,可变数据类型更改地址不发生改变。 -
示例
以比较常见的 String为immutable例子:String s = "a"; s = s.concat("b");
在完成第二行代码后的 s 和 第一行代码的 s 不是同一个对象,等于 JVM 在第二行为你创建了一个新的对象并进行赋值,原来对象"a"并未改变且暂时存在,变化的只是s的引用。以StringBuilder 为mutable例子:
StringBuilder sb = new StringBuilder("a"); sb.append("b");
在完成第一行和第二行后,sb 的这个对象引用没有改变,但其中记录的值发生了变化。 -
使用注意
从ADT设计的角度出发,需要关注属性的可变性与不可变性与方法的防御式编程。如果要设计一个immutable的类,就必须对rep的各属性有清晰认知,如果有mutable类型作为rep,就必须进行防御式拷贝。同时类里不能有mutator方法。
例如设计如下ADT时,HashSet和Calendar都是mutable类型,而要求ADT是表示不变的。//immutable public class Vote<C> { private Set<VoteItem<C>> voteItems = new HashSet<>(); private Calendar date = Calendar.getInstance(); ``````
设计时应该遵循如下规则:
*声明所有的成员变量是private和final的(更安全)
*构造函数对输入进行防御式拷贝。
*在返回mutable类型的方法后面同样进行防御式拷贝(防止表示泄露)。
*不提供任何mutator方法。
总结:
1.不可变数据类型更改后地址发生改变,可变数据类型更改地址不发生改变
2.对可变类型可能造成的风险,我们通过防御式拷贝,返回一个与原地址不相同的可变类型的对象
3.使用可变数据类型可获得更好的性能 ;不可变类型更“安全”但需要临时拷贝和垃圾回收。