不变模式

不变模式
不变模式(immutable pattern)

一个类的内部状态创建后,在整个生命期间都不会发生变化时,就是不变类。这种使用不变类的做法叫做不变模式。

不变模式有两种形式:一种是弱不变模式,另一种是强不变模式,比如String类 math类,Integer类都是强不变类。


弱不变模式:

一个类的实例的状态是不可变化的,但是这个类的引用的实例具有可能会变化的状态。这样的类符合弱不变模式的定义。要实现弱不变模式,一个类必须满足如下条件:


    第一,对象没有任何方法会修改对象的状态,当对象的构造函数对对象的状态初始化之后,对象的状态便不再改变。


    第二,所有的属性都应当是私有的,以防客户端对象直接修改任何的内部状态。


    第三,这个对象所引用的对象如果是可变对象的话,必须设法限制外界对这个对象的访问,以防止对这些对象的修改。如果可能应该尽量在不变对象的内部来初始化。


   
弱不变类的缺点:


弱不变类的子类的对象可以是可变的,这削弱了弱不变类的作用。
可变的子类的对象可能可能修改父对象的状态,有机会允许外界修改父对象的状态,破坏了不变性。

代码演示:

    /**
     * @author chenyu
     *
     */
    public class User {
     
        private String name;
     
        public String getName() {
            return name;
        }
     
        public void setName(String name) {
            this.name = name;
        }
     
    }
     
    /**
     * 弱不变模式
     *
     * @author chenyu
     *
     */
    public class WeakImmutable {
     
        // 属性私有,满足条件2
        private int state;
        // 属性私有,满足条件2
        private User user;
     
        private Integer age;
     
        public WeakImmutable(int state, User user, Integer age) {
            this.state = state;
            this.user = user;
            this.age = age;
        }
     
        public int getState() {
            return this.state;
        }
     
        public User getUser() {
            return this.user;
        }
     
        public Integer getAge() {
            return this.age;
        }
     
        public void setState() {
            // 对象没有任何方法修改对象的状态,满足条件1
            // do nothing.
        }
     
        public static void main(String[] args) {
            int state = 0;
            User u = new User();
            Integer age = 100;
            u.setName("yes");
            WeakImmutable weak = new WeakImmutable(state, u, age);
            System.out.println("原始值:" + weak.getState() + ","
                    + weak.getUser().getName() + "," + weak.getAge());
            // 修改引用后
            state = 5;
            // User由于是可变对象引用,所以有影响
            u.setName("no");
            age = 200;
            System.out.println("修改引用后:" + weak.getState() + ","
                    + weak.getUser().getName() + "," + weak.getAge());
        }
    }


结果:可以看到user的名字会改变。

原始值:0,yes,100
修改引用后:0,no,100

强不变模式:


    一个类的实例的状态不会改变,同时它的子类的实例也具有不可变化的状态。这样的类符合强不变模式。要实现强不变模式,一个类必须首先满足弱不变模式所要求的所有条件,并且还要满足下面条件之一:

    第一,所考虑的类所有的方法都应当是final,这样这个类的子类不能够置换掉此类的方法。
    第二,这个类本身就是final的,那么这个类就不可能会有子类,从而也就不可能有被子类修改的问题。


不变模式在Java中的应用
如String类
String a = "123" ;
String a1 = "123" ;
String a2 = "123" ;
String a3 = "1234" ;

java虚拟机只会创建一个字符串实例,a,a1,a2对象共享一个值。遇到不同的字符串,java虚拟机会再创建一个String对象,如a3。如果程序所处理的字串有频繁的内容变化,就不宜使用String类型,而应当使用StringBuffer类型,如果需要对字串做大量的循环查询,也不宜使用String类型,应当考虑使用byte或char数组,math这个类也是不变类,你可以去看看源码。


其它不变类:
The Integer,String, Float, Double, Byte, Long, Short, Boolean, and Character classes are all examples of an immutable class. By definition, you may not alter the value of an immutable object after its construction.In Java, a class such as Integer acts as a simple wrapper around its primitive counterpart -- in this case, int. The wrappers found in java.lang allow us to treat the primitives as if they were objects. So, for example, you could not put an int into a Vector without wrapping it。


优缺点:
不变模式可增强对象的健壮性。不变模式允许多个对象共享某一对象,降低了对该对象进行并发访问时的同步化开销。唯一缺点是一旦需要修改一个不变对象的状态,就只好创建一个新的同类对象,在需要频繁修改不变对象的环境里,会有大量的不变对象作为中间结果被创建出来,再被Java的垃圾收集器收走,这是一种资源的浪费。




 




作者:chen.yu
深信服三年半工作经验,目前就职游戏厂商,希望能和大家交流和学习,
微信公众号:编程入门到秃头 或扫描下面二维码
零基础入门进阶人工智能(链接)