http://www.iteye.com/topic/959751

最近老有人问我不变模式,我其实也理解得不深,于是花了一些时间进行学习总结,分析了一下不变模式(immutable pattern),和大家一起分享。说得不对的地方欢迎拍砖,谢绝谩骂。姐妹篇:精通有状态vs无状态(Stateful vs Stateless).http://www.iteye.com/topic/960532 不变模式(immutable pattern)
一个类的内部状态创建后,在整个生命期间都不会发生变化时,就是不变类。这种使用不变类的做法叫做不变模式。
不变模式有两种形式:一种是弱不变模式,另一种是强不变模式。
弱不变模式: 一个类的实例的状态是不可变化的,但是这个类的引用的实例具有可能会变化的状态。这样的类符合弱不变模式的定义。要实现弱不变模式,一个类必须满足如下条件:
    第一,对象没有任何方法会修改对象的状态,当对象的构造函数对对象的状态初始化之后,对象的状态便不再改变。
    第二,所有的属性都应当是私有的,以防客户端对象直接修改任何的内部状态。
    第三,这个对象所引用的对象如果是可变对象的话,必须设法限制外界对这个对象的访问,以防止对这些对象的修改。如果可能应该尽量在不变对象的内部来初始化。
    弱不变模式的缺点是: 一个弱不变对象引用的实例变量可以是可变对象,可能会通过外界修改父对象的状态,这是一个显著的缺点。可以在初始化可变对象时,先进行clone。
代码演示:

  1. /**
  2. * @author Peter Wei
  3. *
  4. */
  5. public class User {
  6. private String name;
  7. public String getName() {
  8. return name;
  9. }
  10. public void setName(String name) {
  11. this.name = name;
  12. }
  13. }
  14. /**
  15. * 弱不变模式
  16. *
  17. * @author Peter Wei
  18. *
  19. */
  20. public class WeakImmutable {
  21. // 属性私有,满足条件2
  22. private int state;
  23. // 属性私有,满足条件2
  24. private User user;
  25. private Integer age;
  26. public WeakImmutable(int state, User user, Integer age) {
  27. this.state = state;
  28. this.user = user;
  29. this.age = age;
  30. }
  31. public int getState() {
  32. return this.state;
  33. }
  34. public User getUser() {
  35. return this.user;
  36. }
  37. public Integer getAge() {
  38. return this.age;
  39. }
  40. public void setState() {
  41. // 对象没有任何方法修改对象的状态,满足条件1
  42. // do nothing.
  43. }
  44. public static void main(String[] args) {
  45. int state = 0;
  46. User u = new User();
  47. Integer age = 100;
  48. u.setName("yes");
  49. WeakImmutable weak = new WeakImmutable(state, u, age);
  50. System.out.println("原始值:" + weak.getState() + ","
  51. + weak.getUser().getName() + "," + weak.getAge());
  52. // 修改引用后
  53. state = 5;
  54. // User由于是可变对象引用,所以有影响
  55. u.setName("no");
  56. age = 200;
  57. System.out.println("修改引用后:" + weak.getState() + ","
  58. + weak.getUser().getName() + "," + weak.getAge());
  59. }
  60. }
/**
* @author Peter Wei
*
*/
public class User { private String name; public String getName() {
return name;
} public void setName(String name) {
this.name = name;
} } /**
* 弱不变模式
*
* @author Peter Wei
*
*/
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
我们再引伸一个不可变类的例子: 在时间截止时,我们需要一一检查队列成员是不是vip,如果是可以去USA.假设是多线程环境,并且users数组是多线程共享,那么另外的线程通过users去修改users[n],这时就会把users[n]绕过时间检查而去USA.

  1. /**
  2. * 不变模式之clone
  3. *
  4. * @author Peter Wei
  5. *
  6. */
  7. public class WeakImmutableClone {
  8. public static void main(String[] args) {
  9. User[] users = new User[3];
  10. users[0] = new User();
  11. users[0].setName("peterwei");
  12. users[1] = new User();
  13. users[1].setName("Tomssssss");
  14. users[2] = new User();
  15. users[2].setName("peterwei88");
  16. time4Check();
  17. /*
  18. * 时间到,我们需要一一检查队列成员是不是vip,如果是可以去USA.假设是多线程环境,并且users数组是多线程共享,
  19. * 那么另外的线程通过users去修改users[n],这时就会把users[n]绕过时间检查而去USA.
  20. */
  21. goUSA(users);
  22. }
  23. public static void goUSA(User[] users) {
  24. // User[] tmp = new User[users.length];
  25. // System.arraycopy(users, 0, tmp, 0, users.length);
  26. for (User u : users) {
  27. if (checkVip(u)) {
  28. System.out.println("You can go!");
  29. } else {
  30. System.out.println("go away!");
  31. }
  32. }
  33. }
  34. public static boolean checkVip(User user) {
  35. if (user.getName().startsWith("peterwei")) {
  36. return true;
  37. }
  38. return false;
  39. }
  40. public static void time4Check() {
  41. // 假设时间期限到,要检查上万人以上的队列。
  42. }
  43. }
/**
* 不变模式之clone
*
* @author Peter Wei
*
*/
public class WeakImmutableClone { public static void main(String[] args) { User[] users = new User[3];
users[0] = new User();
users[0].setName("peterwei");
users[1] = new User();
users[1].setName("Tomssssss");
users[2] = new User();
users[2].setName("peterwei88"); time4Check();
/*
* 时间到,我们需要一一检查队列成员是不是vip,如果是可以去USA.假设是多线程环境,并且users数组是多线程共享,
* 那么另外的线程通过users去修改users[n],这时就会把users[n]绕过时间检查而去USA.
*/
goUSA(users); } public static void goUSA(User[] users) { // User[] tmp = new User[users.length];
// System.arraycopy(users, 0, tmp, 0, users.length); for (User u : users) {
if (checkVip(u)) {
System.out.println("You can go!");
} else {
System.out.println("go away!");
} }
} public static boolean checkVip(User user) {
if (user.getName().startsWith("peterwei")) {
return true;
}
return false;
} public static void time4Check() {
// 假设时间期限到,要检查上万人以上的队列。
}
}

解决方法: 在事务处理及数据大批量入库的多线程环境中,应该也会有类似的问题。所以对于这样的传入参数及上例中的不变对象引用可变对象,我们可以将其在相关构造函数及方法中复制为本地变量(数组),及使用它的深度clone,阻止相关数据与外部线程的联系。

  1. public static void goUSA(User[] users) {
  2. User[] tmp = new User[users.length];
  3. System.arraycopy(users, 0, tmp, 0, users.length);
  4. for (User u : tmp) {
  5. if (checkVip(u)) {
  6. System.out.println("You can go!");
  7. } else {
  8. System.out.println("go away!");
  9. }
  10. }
  11. }
	public static void goUSA(User[] users) {

		User[] tmp = new User[users.length];
System.arraycopy(users, 0, tmp, 0, users.length); for (User u : tmp) {
if (checkVip(u)) {
System.out.println("You can go!");
} else {
System.out.println("go away!");
} }
}

强不变模式:
    一个类的实例的状态不会改变,同时它的子类的实例也具有不可变化的状态。这样的类符合强不变模式。要实现强不变模式,一个类必须首先满足弱不变模式所要求的所有条件,并且还要满足下面条件之一:     第一,所考虑的类所有的方法都应当是final,这样这个类的子类不能够置换掉此类的方法。     第二,这个类本身就是final的,那么这个类就不可能会有子类,从而也就不可能有被子类修改的问题。
不变模式在Java中的应用 如String类

  1. String a = "123" ;
  2. String a1 = "123" ;
  3. String a2 = "123" ;
  4. String a3 = "1234" ;
		String a = "123" ;
String a1 = "123" ;
String a2 = "123" ;
String a3 = "1234" ;

java虚拟机只会创建一个字符串实例,a,a1,a2对象共享一个值。遇到不同的字符串,java虚拟机会再创建一个String对象,如a3。如果程序所处理的字串有频繁的内容变化,就不宜使用String类型,而应当使用StringBuffer类型,如果需要对字串做大量的循环查询,也不宜使用String类型,应当考虑使用byte或char数组. 
其它不变类: 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的垃圾收集器收走,这是一种资源的浪费。
总结: 不变模式的核心就是对象不变,从而引伸出对象复用共享的思想。如无状态的单例模式,享元(Flyweight)模式及原型模式(Prototype)都可以认为是不变模式的应用。其它如线程池,缓存等的实现也一定程度上是使用不变模式。还有EJB的Stateless Session Bean(无状态会话bean),Spring对Service层、Dao层bean的默认单例实现,我认为都是沿用了不变模式中共享的思想。

http://www.cnblogs.com/afarmer/archive/2012/03/12/2392363.html

设计模式之不变模式(Immutable Pattern)分析的更多相关文章

  1. 多线程程序设计学习(3)immutable pattern模式

    Immutable pattern[坚不可摧模式] 一:immutable pattern的参与者--->immutable(不变的)参与者        1.1:immutable参与者是一个 ...

  2. 乐在其中设计模式(C#) - 观察者模式(Observer Pattern)

    原文:乐在其中设计模式(C#) - 观察者模式(Observer Pattern) [索引页][源码下载] 乐在其中设计模式(C#) - 观察者模式(Observer Pattern) 作者:weba ...

  3. 乐在其中设计模式(C#) - 装饰模式(Decorator Pattern)

    原文:乐在其中设计模式(C#) - 装饰模式(Decorator Pattern) [索引页][源码下载] 乐在其中设计模式(C#) - 装饰模式(Decorator Pattern) 作者:weba ...

  4. 乐在其中设计模式(C#) - 适配器模式(Adapter Pattern)

    原文:乐在其中设计模式(C#) - 适配器模式(Adapter Pattern) [索引页][源码下载] 乐在其中设计模式(C#) - 适配器模式(Adapter Pattern) 作者:webabc ...

  5. 乐在其中设计模式(C#) - 单例模式(Singleton Pattern)

    原文:乐在其中设计模式(C#) - 单例模式(Singleton Pattern) [索引页][源码下载] 乐在其中设计模式(C#) - 单例模式(Singleton Pattern) 作者:weba ...

  6. 多线程学习之二坚不可摧模式Immutable pattern

    Immutable pattern[坚不可摧模式] 一:immutable pattern的参与者--->immutable(不变的)参与者        1.1:immutable参与者是一个 ...

  7. 怎样让孩子爱上设计模式 —— 7.适配器模式(Adapter Pattern)

    怎样让孩子爱上设计模式 -- 7.适配器模式(Adapter Pattern) 标签: 设计模式初涉 概念相关 定义: 适配器模式把一个类的接口变换成client所期待的还有一种接口,从而 使原本因接 ...

  8. 说说设计模式~大话目录(Design Pattern)

    回到占占推荐博客索引 设计模式(Design pattern)与其它知识不同,它没有华丽的外表,没有吸引人的工具去实现,它是一种心法,一种内功,如果你希望在软件开发领域有一种新的突破,一个质的飞越,那 ...

  9. 深入浅出设计模式——观察者模式(Observer Pattern)

    模式动机 建立一种对象与对象之间的依赖关系,一个对象发生改变时将自动通知其他对象,其他对象将相应做出反应.在此,发生改变的对象称为观察目标,而被通知的对象称为观察者,一个观察目标可以对应多个观察者,而 ...

随机推荐

  1. ClassNotFoundException: http.nio.NHttpClientEventHandle

    解决方案是打开maven仓库中的jar包看看报错的类所对应版本的类存在不存在,若不存在就换个版本的jar包

  2. 孤荷凌寒自学python第二十九天python的datetime.time模块

     孤荷凌寒自学python第二十九天python的datetime.time模块 (完整学习过程屏幕记录视频地址在文末,手写笔记在文末) datetime.time模块是专门用来表示纯时间部分的类. ...

  3. (原)Unreal 渲染模块 渲染流程

    @author:白袍小道 浏览分享随缘,评论不喷亦可.     扯淡部分: 在temp中,乱七八糟的说了下大致的UE过程.下面我们还是稍微别那么任性,一步步来吧.     UE渲染模块牵扯到场景遍历. ...

  4. [转载]kd tree

    [本文转自]http://www.cnblogs.com/eyeszjwang/articles/2429382.html k-d树(k-dimensional树的简称),是一种分割k维数据空间的数据 ...

  5. django的F和Q对象

    F表达式和Q表达式: # 示例模型如下 class Book(models.Model): """图书模型""" name = models ...

  6. Limeng:Individual Project: Word frequency program -BUAA Advanced Software Engineering

    11061190-李孟 Implement a console application to tally the frequency of words under a directory (2 mod ...

  7. 201621123033 《Java程序设计》第6周学习总结

    第六次作业 1. 本周学习总结 1.1 面向对象学习暂告一段落,请使用思维导图,以封装.继承.多态为核心概念画一张思维导图或相关笔记,对面向对象思想进行一个总结. 2. 书面作业 1. clone方法 ...

  8. [洛谷P3857][TJOI2008]彩灯

    题目大意:有$n$盏灯,$m$个开关($n,m\leqslant 50$),每个开关可以控制的灯用一串$OX$串表示,$O$表示可以控制(即按一下,灯的状态改变),$X$表示不可以控制,问有多少种灯的 ...

  9. uoj169:元旦老人与数列

    题意:http://uoj.ac/problem/169 sol  :线段树..........蜜汁TLE了一个点,不管了..... 代码抄snowMyDream的,orz........... 线段 ...

  10. .net使用memcached

    Windows中memached安装 -------------服务器端配置 1>开始>运行:CMD(确定) 2>cd C:\memcached(回车) 3>memcached ...