在多线程环境中,为了保证共享数据的一致性,往往需要对共享数据的使用进行加锁,但是加锁操作本身就会带来一定的开销,这里可以使用将共享数据使用不可变对象进行封装,从而避免加锁操作。

1. 模型角色

       不可变对象指的是,对象内部没有提供任何可供修改对象数据的方法,如果需要修改共享变量的任何数据,都需要先构建整个共享对象,然后对共享对象进行整体的替换,通过这种方式来达到对共享对象数据一致性的保证。如下是不可变对象设计的类图:

如下是各个角色功能的描述:

  • ImmutableObject:不可变对象的载体。对于需要一致性更改的数据,都需要放入不可变对象中,对于不可变对象,需要注意如下几点:

    • 不可变对象的属性必须使用final修饰,以防止属性被意外修改,并且final还可以保证JVM在该对象构造完成时该属性已经初始化成功(JVM在构造完对象时可能只是为其分配了引用空间,而各个属性值可能还未初始化完成);
    • 属性的设值必须在构造方法中统一构造完成,其余的方法只是提供的查询各个属性相关的方法;
    • 对于可变状态的引用类型属性,如集合,在获取该类型的属性时,必须返回该属性的一个深度复制结果,以防止不可变对象的该属性值被客户端修改;
    • 不可变对象的类必须使用final修饰,以防止子类对其本身或其方法进行修改;
  • Manipulator:聚合对象的管理类(某些情况可不用)。对于聚合对象的管理,主要有两部分:查询和修改。对于聚合对象的查询,只需要根据一定的规则在Manipulator类中获取该对象即可,对于聚合对象的修改,需要首先通过参数构造一个完整的聚合对象,然后将保存的该聚合对象的引用进行替换即可;
  • Client:获取聚合对象的客户端应用。

2. 使用场景

       对于不可变对象,其主要有如下三种使用场景:

  • 当某组数据变化不是很频繁,则可以使用不可变对象。对于数据的访问,由于不可变对象的引用空间不会发生变化,因而任何线程都可以保有同一个不可变对象的引用,这样可以减少内存的消耗,也能保证数据访问的一致性;
  • 当某组数据需要进行一致性的更新操作时,可以使用不可变对象。由于不可变对象能够保证对其任何数据的修改都是对整个对象的替换,因而其能够保证整组数据的一致性。需要注意的是,如果该组数据变更比较频繁,则不宜使用不可变对象,因为这会造成创建大量的不可变对象,从而增加JVM垃圾回收的压力。具体的情况应根据JVM可使用内存大小与对象更新的频率进行考量;
  • 当需要对象作为Map的键时可以使用不可变对象。对于Map而言,其hashCode()方法默认返回的是对象的引用地址,而对于不可变对象而言,由于其引用地址是不会发生变化的,因而即使不对其hashCode()方法进行重写,其也不会发生变化。

3. 使用示例

       对于不可变对象,一个很好的例子就是地址经纬度。笔者所工作的公司处理的业务和房源相关,其中有一部分就是需要处理房源所在点的经纬度信息,这里就可以使用不可变对象,因为房源经纬度基本上不会发生变化,并且对其操作也主要是以查询为主,最重要的是,对经纬度的处理必须是经度和纬度同时发生变化,任何情况下只更改了其中一个数据都会产生问题。如下是记录房源经纬度的类:

public final class Location {
private final long id;
private final String latitude;
private final String longitude; public Location(long id, String latitude, String longitude) {
this.id = id;
this.latitude = latitude;
this.longitude = longitude;
} public long getId() {
return id;
} public String getLatitude() {
return latitude;
} public String getLongitude() {
return longitude;
}
}

       该Location类也即上述UML类图中的ImmutableObject部分。可以看到,任何对Location对象的修改都必须重新构建一个Location对象。如下是对Location的管理类,用于存储具体的Location信息的:

public class LocationHolder {

  private final LocationHolder INSTANCE = new LocationHolder();
private Map<Long, Location> locations; private LocationHolder() {
this.locations = new ConcurrentHashMap<>();
} public LocationHolder getInstance() {
return INSTANCE;
} public Location getLocation(long id) {
return locations.get(id);
} public void addLocation(long id, String latitude, String longitude) {
Location location = new Location(id, latitude, longitude);
locations.put(id, location);
} public Map<Long, Location> getLocations() {
return Collections.unmodifiableMap(locations);
}
}

        可以看到,这里对Location的管理是通过一个单例类LocationHolder进行的,任何对Location的操作都进行了封装,并且这里批量获取Location,也是返回了一个不可变Map,从而保证原始数据不会作任何修改,如果该Map的键或值任何一方可能发生变化,那么在返回值则必须返回一个深度复制的结果,这样才能保证原始数据的完整性。

Java多线程编程之不可变对象模式的更多相关文章

  1. Java多线程编程(二)对象及变量的并发访问

    一.synchronized同步方法 1.方法内的变量为线程安全 “非线程安全”问题存在于“实例变量”中,如果是方法内部的私有变量,则不存在“非线程安全”问题,所得结果也就是“线程安全”的了. 示例: ...

  2. java多线程编程模式

    前言 区别于java设计模式,下面介绍的是在多线程场景下,如何设计出合理的思路. 不可变对象模式 场景 1. 对象的变化频率不高 每一次变化就是一次深拷贝,会影响cpu以及gc,如果频繁操作会影响性能 ...

  3. Java多线程编程模式实战指南(二):Immutable Object模式--转载

    本文由本人首次发布在infoq中文站上:http://www.infoq.com/cn/articles/java-multithreaded-programming-mode-immutable-o ...

  4. Java多线程编程核心技术---对象及变量的并发访问(二)

    数据类型String的常量池特性 在JVM中具有String常量池缓存的功能. public class Service { public static void print(String str){ ...

  5. Java多线程编程中Future模式的详解

    Java多线程编程中,常用的多线程设计模式包括:Future模式.Master-Worker模式.Guarded Suspeionsion模式.不变模式和生产者-消费者模式等.这篇文章主要讲述Futu ...

  6. Java多线程编程核心技术(二)对象及变量的并发访问

    本文主要介绍Java多线程中的同步,也就是如何在Java语言中写出线程安全的程序,如何在Java语言中解决非线程安全的相关问题.阅读本文应该着重掌握如下技术点: synchronized对象监视器为O ...

  7. Java多线程编程中Future模式的详解<转>

    Java多线程编程中,常用的多线程设计模式包括:Future模式.Master-Worker模式.Guarded Suspeionsion模式.不变模式和生产者-消费者模式等.这篇文章主要讲述Futu ...

  8. Java多线程编程详解

    转自:http://programming.iteye.com/blog/158568 线程的同步 由于同一进程的多个线程共享同一片存储空间,在带来方便的同时,也带来了访问冲突这个严重的问题.Ja ...

  9. Java多线程编程核心技术

    Java多线程编程核心技术 这本书有利于对Java多线程API的理解,但不容易从中总结规律. JDK文档 1. Thread类 部分源码: public class Thread implements ...

随机推荐

  1. P2434 [SDOI2005]区间

    题目描述 现给定n个闭区间[ai, bi],1<=i<=n.这些区间的并可以表示为一些不相交的闭区间的并.你的任务就是在这些表示方式中找出包含最少区间的方案.你的输出应该按照区间的升序排列 ...

  2. asp.net core合并压缩资源文件(转载)

    在asp.net core中使用BuildBundlerMinifier合并压缩资源文件 在asp.net mvc中可以使用Bundle来压缩合并css,js 不知道的见:http://www.cnb ...

  3. CentOS 安装第三方yum源

    yum install wget #安装下载工具 wget http://www.atomicorp.com/installers/atomic #下载 sh ./atomic #安装 yum che ...

  4. Animate.css介绍

    Animate.css简介 animate.css 动画库,预设了抖动(shake).闪烁(flash).弹跳(bounce).翻转(flip).旋转(rotateIn/rotateOut).淡入淡出 ...

  5. iOS开发网络篇—发送GET和POST请求(使用NSURLSession) - 转

    说明: 1.该文主要介绍如何使用NSURLSession来发送GET请求和POST请求 2.本文将不再讲解NSURLConnection的使用,如有需要了解NSURLConnection如何发送请求. ...

  6. java常用API之Date类

    Date类: 类 Date 表示特定的瞬间,精确到毫秒. 毫秒概念:1000毫秒=1秒 毫秒的0点: System.currentTimeMillis()  返回值long类型参数   用于获取当前日 ...

  7. Unix中Signal信号的不同

    Unix系统signal函数的不同 (1)函数说明 在signal函数中,有两个形参,分别代表需要处理的信号编号值和处理信号函数的指针.它主要是用于前32种非实时信号的处理,不支持信号的传递信息.但是 ...

  8. 打开CDQ的大门&BZOJ3262

    题目传送门 第一次接触CDQ分治,感谢YZ大佬的教导. CDQ分治就是一种奇特的分治方法,它用左区间的区间信息来更新右区间. 设CDQ(L,R,l,r)表示递归到区间[L,R],区间的值为[l,r]. ...

  9. 【HNOI2014】江南乐

    题面 题解 知识引入 - \(SG\)函数 任何一个公平组合游戏都可以通过把每个局面看成一个顶点,对每个局面和它的子局面连一条有向边来抽象成这个"有向图游戏".下面我们就在有向无环 ...

  10. 【Vijos】lxhgww的奇思妙想

    题面 题解 求$k$级祖先孙子 为什么要用长链剖分啊??? 倍增并没有慢多少... 其实是我不会 长链剖分做这道题还是看这位巨佬的吧. 代码 #include<bits/stdc++.h> ...