“同步”确保了操作的原子性执行,但它还有其它重要的方面:memory visibility.我们不但要确保当一个线程在使用一个对象的时候,其它线程不能修改这个对象,而且还要保证该线程在修改对象状态时,其它线程能够看到该线程对对象所做的改变。

可以通过显式的同步语句或内建类库的同步机制以保证对象的正确发布。

3.1. Visibility

3.1.1. Stale Data

3.1.2. Non-atomic 64-bit Operations(针对没有声明为volatile的double和long)

Java内存模型要求读取和写入操作都是原子性的,但对于非volitile的long和double变量,jvm通过两个32位操作来完成一个64位的读写。

因此,当在一个多线程程序中有double和long型的共享变量是,对它的读写一定要加volatile或通过锁来同步

3.1.3. Locking and Visibility

3.1.4. Volatile Variables

锁能确保可见性和原子性,volitale只能保证可见性

3.2. Publication and Escape

发布一个对象是指该对象在它当前的上下文范围之外是可用的,如存储一个引用,使其他代码能够访问、通过非私有的方法返回自己、或者是将自己传递给另一个类的方法中(特别注意!!)。

//Publishing an Object
//发布一个对象,通过静态字段引用它
public static Set<Secret> knownSecrets; public void initialize(){
knownSecrets = new HashSet<Secret>();
}
//Allowing Internal Mutable State to Escape.Don't do this.
//私有变量通过公共方法逃逸 class UnsafeStates{
private String[] states = new String[]{"ak","al",...}; public String[] getStates(){
return states;
}
}

发布一个对象,有可能同时会发布它字段锁引用的对象,这些对象通过它的公有变量或共有的get方法发布出来,要小心无意的泄漏。

小心,不要在你的构造方法里面泄漏了你的对象!!!

public class ThisEscape{
public ThisEscape(EventSource source){
source.registerListener(new EventListener{
public void onEvent(Event e){
doSomething(e);
}
});
}
}

3.2.1. Safe Construction Practices

从构造函数中发布一个对象(匿名内部类实现),会导致当前对象也跟着发布,但是当前对象的构造函数还没有返回,即它不一定被正确的构建就已经发布了

Do not allow the this reference to escape during construction.

一个常见错误是在构造函数中起一个线程,这样会提前发布this引用。

//ThisEscape 改进版本
public class SafeListener{
private final EventListener listener; private SafeListener(){
listener = new EventListener(){
public void onEvent(Event e){
doSomething(e);
}
};
} public static SafeListener newInstance(EventSource source){
SafeListener safe = new SafeListener();
source.registerListener(safe.listener);
return safe;
}
}

3.3. Thread Confinement

3.3.1. Ad-hoc Thread Confinement

3.3.2. Stack Confinement

就是本地变量

//本地原是类型和引用类型的线程封闭
public int loadTheArk(Collection<Animal> candidates){
SortedSet<Animals> animals;
int numPairs = 0;
Animal candidate = null;
animals = new TreeSet<Animal>(new SpeciesGenderComparator());
animals.addAll(candidates);
for(Animal a:animals){
if(candidate == null || !candidate.isPotentialMate(a))
candidate = a;
esle{
ark.load(new AnimalPair(candidate,a));
++ numPairs;
candidates = null;
}
}
return numPairs;
}

3.3.3. ThreadLocal

//using ThreadLocal to ensure thread confinement
private static ThreadLocal<Connection> connectionHolder = new ThreadLocal<Connection>(){
public Connection initialValue(){
return DriverManager.getConnection(dburl);
}
}; public static Connection getConnection(){
return connectionHolder.get();
}

该技术适用于频繁申请临时对象的操作,即每来一个线程,就分配一个临时对象的情况。但是不要滥用,会增加耦合性。

3.4. Immutability

不可变对象总是线程安全的。

//一个不可变对象
@Immutable
public final class /threeStooges{
private fianl Set<String> stooges = new HashSet<String>(); public ThreeStooges(){
stooges.add("Moe");
atooges.add("Larry");
stooges.add("Curly");
}
public boolean isStooge(String name){
return stooges.contain(name);
}
}

3.4.1 Final Fields

就像对象的字段如果不需要保持对外可见,用private修饰一样,若字段不需要修改,那么也要设置为final

3.4.2 Example: Using Volatile to Publish Immutable Objects

3.5. Safe Publication

3.5.1 Improper Publication:When Good Objects Go Bad

3.5.2 Immutable Objects and initialization Safety

3.5.3 Safe Publication Idioms

要安全发布一个对象,对该对象的引用与对象的状态必须同时对其他线程可见。一个正确构建的对象可以通过如下步骤被安全发布:

  • 使用静态初始化器初始化一个对象引用
  • 将引用存储为volatile或AtomicReference
  • 将引用存为一个正确构建的对象的final字段
  • 将引用存为一个字段,该字段用适当的锁来保护

推荐使用静态初始化器,如:

public static Holder holder = new Holder(12);

这样发布的对象一定是安全的。

3.5.4 Efectively immutable Objects

事实不可变对象(对象在发布后就不需要改变,但是技术上是可变的),在发布之后不会被修改的对象,那么只要保证安全发布就好了,由于后续不需修改,也就不用同步机制来保证可见性。例如,Date就是可变对象,但是假如你不去修改它,那么就不需要锁来保证可见性。

3.5.5 Mutable Objects

对象的发布需求取决于它的可变性:

  • 不可变对象可以通过任何机制发布
  • 事实不可变对象必须安全发布
  • 可变对象必须安全发布,而且必须线程安全或通过锁来保护

3.5.6 Sharing Objects Safely

当你发布一个对象,你要指定该对象应该被怎样访问

在并发程序中,使用和共享对象的最有效的策略是:

  • 线程封闭:该对象为该线程独有,其他线程无法修改
  • 只读共享:其它线程无法修改该对象,也就不会有线程问题,可以用不可变对象或事实不可变对象
  • 线程安全共享:一个线程安全的对象使用内部同步,那么多线程可以自由的获取或修改它而不需要额外的同步机制
  • 监视状态:一个受监视的对象只能通过持有正确锁的线程来访问。

JCIP chap3 share objects的更多相关文章

  1. The Truth About .NET Objects And Sharing Them Between AppDomains

    From http://geekswithblogs.net/akraus1/archive/2012/07/25/150301.aspx I have written already some ti ...

  2. Managing C++ Objects: 管理C++对象 —— 一些建议准则

    原文链接: Managing C++ Objects Here are some guidelines I have found useful for writing C++ classes. The ...

  3. AMD高级应用(翻译)

    Dojo now supports modules written in the Asynchronous Module Definition (AMD) format, which makes co ...

  4. 企业级nosql数据库应用与实战-redis

    一.NoSQL简介 1.1 常见的优化思路和方向 1.1.1 MySQL主从读写分离 由于数据库的写入压力增加,Memcached只能缓解数据库的读取压力.读写集中在一个数据库上让数据库不堪重负,大部 ...

  5. Python 函数参数传递机制.

    learning python,5e中讲到.Python的函数参数传递机制是对象引用. Arguments are passed by assignment (object reference). I ...

  6. 超越村后端开发(4:API开发)

    1.users相关的api开发 1.在settings中添加APPID,SECRET 2.在apps/users/views.py内: from chaoyuecun.settings import ...

  7. 设计模式教程(Design Patterns Tutorial)笔记之二 结构型模式(Structural Patterns)

    目录 · Decorator · What is the Decorator Design Pattern? · Sample Code · Adapter · What is the Adapter ...

  8. facebook api之Access Tokens之Business Manager System User

    Business Manager System User Make programatic, automated actions on ad objects or Pages, or do progr ...

  9. [Forward]Ten Caching Mistakes that Break your App

    Caching large objects, duplicate objects, caching collections, live objects, thread unsafe caching a ...

随机推荐

  1. 阿里云系列——7.阿里云IIS系列详解(过程+通用+最新)

    网站部署之~阿里云系列汇总 http://www.cnblogs.com/dunitian/p/4958462.html 先讲IIS系列,Linux部署以后再继续讲 先打开主机管理平台,确认域名绑定 ...

  2. SQLServer:什么是主键(PK)和外键(FK)?

    一.主键与外键 1.主键是用来唯一地标识一行数据.主键列必须包含唯一的值,且不能包含空值(null). 2.主键可以建立在每张二维表中单列或者多列上. 3.一张二维表上的外键可以引用另一张二维表上对应 ...

  3. Java多线程系列--“JUC锁”04之 公平锁(二)

    概要 前面一章,我们学习了“公平锁”获取锁的详细流程:这里,我们再来看看“公平锁”释放锁的过程.内容包括:参考代码释放公平锁(基于JDK1.7.0_40) “公平锁”的获取过程请参考“Java多线程系 ...

  4. MySQL笔记---视图,存储过程, 触发器的使用入门

    大二学数据库的时候,只是隐约听到老师提起过视图啊,存储过程啊,触发器啊什么的,但只是淡淡的记住了名字,后来自己做些小项目,小程序,也没有用上过,都只是简单的建表,关联表之类的,导致我对这些东西的理解只 ...

  5. 使用OAuth、Identity创建WebApi认证接口供客户端调用

    前言 现在的web app基本上都是前后端分离,之前接触的大部分应用场景最终产品都是部署在同一个站点下,那么随着WebApi(Restful api)的发展前后端实现的完全分离,前端不在后端框架的页面 ...

  6. 如何在Windows Server 2008 R2没有磁盘清理工具的情况下使用系统提供的磁盘清理工具

    今天,刚好碰到服务器C盘空间满的情况,首先处理了临时文件和有关的日志文件后空间还是不够用,我知道清理C盘的方法有很多,但今天只分享一下如何在Windows Server 2008 R2没有磁盘清理工具 ...

  7. react-native 简单的导航

    默默潜水了两年了,一直都在看大神们写的博客,现在我也分享一下跟RN导航有关的东西. 前两年我主要是做iOS开发的,现在刚找了份工作,应公司要求,现在开始学习reactnative的东西,由于我以前没怎 ...

  8. [Asp.net 5] Options-配置文件(2)

    很久之前写过一篇介绍Options的文章,2016年再打开发现很多变化.增加了新类,增加OptionMonitor相关的类.今天就对于这个现在所谓的新版本进行介绍. 老版本的传送门([Asp.net ...

  9. 设计模式(八)桥接模式(Bridge Pattern)

    一.引言 这里以电视遥控器的一个例子来引出桥接模式解决的问题,首先,我们每个牌子的电视机都有一个遥控器,此时我们能想到的一个设计是——把遥控器做为一个抽象类,抽象类中提供遥控器的所有实现,其他具体电视 ...

  10. Java语言中的面向对象特性总结

    Java语言中的面向对象特性 (总结得不错) [课前思考]  1. 什么是对象?什么是类?什么是包?什么是接口?什么是内部类?  2. 面向对象编程的特性有哪三个?它们各自又有哪些特性?  3. 你知 ...