java并发--线程安全

处理器的时钟频率已经很难再提高,想要提升计算机的性能,一个很明显的趋势是使用多处理器。
因为程序调度的基本单元是线程,一个单线程程序一次只能运行在一个处理器上,在双处理器系统,就浪费了一般的CPU资源,在100个CPU系统中,就浪费了99%的CPU资源。而使用多线程编程则能够充分利用处理器资源,提高吞吐量。
多线程的使用并不是绝对有利的,它同时也引入了单线程环境不存在的安全性问题,在多线程环境中,因为竞争条件的存在,在没有充分同步的情况下,多线程中的各个操作的顺序是不可预测的,有时甚至让人惊讶。
在设计良好的应用程序使用多线程,能够获得不错的性能收益,但多线程仍会带来一定程度的性能开销。上下文切换,当调度程序时挂起正在运行的线程,另一个线程开始运行--这在多线程系统中是很频繁的,会带来很大的性能消耗;保存和恢复线程执行的上下文,会让CPU的时间花费在对线程的调度而不是运行上。当线程共享数据时,需要使用同步机制,这回限制编译器的优化。
......

什么是线程安全

当多个线程访问一个类时,如果不用考虑这些线程在运行时环境下的调度和交替执行,并且不需要同步及在调用代码时不需要额外的协调,这个类的行为仍是正确的,那么这个类是线程安全的。

无状态的对象永远是线程安全的

public class StatelessServlet implements Servlet{
    public void service(ServletRequest req, ServletResponse res){
        Integer i = extractFromRequest(req);
        Integer[] factors = factor(i);
        encodeIntoResponse(res, factors);
    }
}

以一段伪代码来说明。
StatelessServlet是一个无状态的servlet,因为它不包含域也没有引用其他类的域。线程之间不共享状态,一个请求过来,会唯一地存在本地变量,这些变量保存在线程的栈中,只有执行线程才能访问,不会影响同一个servlet的其他请求线程。
因为线程访问无状态对象的行为,不会影响其他线程访问该对象的正确性,因此无状态对象是线程安全的。

原子性

给上面的伪代码加上计数功能,如下:

public class UnsafeCountServlet implements Servlet{
    private long count = 0; 
    
    public long getCount(){return count;}
    
    public void service(ServletRequest req, ServletResponse res){
        BigInteger i = extractFromRequest(req);
        BigInteger[] factors = factor(i);
        ++count;
        encodeIntoResponse(res, factors);
    }
}

此时的UnsafeCountServlet是一个有状态的类,在多线程环境中不再是线程安全的,因为++count不是一个原子操作,而是三个离散操作的的简写:获取当前值,加1,返回新值,是一个“读-改-写”操作。
这样的操作在多线程环境中很容易出现问题,加入初始值为0,某个时刻线程一读取到值0,此时线程调度,另一个线程二也读到0,然后加1,返回新值1,再切回线程一,加1,返回新值1,这就缺少了一次自增。
出现这样的错误是因为:竞争条件的存在。

竞争条件

当计算的正确性依赖于运行时的时序或者多线程的交替时,就会产生竞争条件。最常见的竞争条件就是“检查再运行”。
下面是一个惰性初始化的例子:

public class LazyInitClass{
    private ExpensiveObject instance = null;
    
    public ExpensiveObject getInstance(){
        if(instance == null){
            instance = new ExpensiveObject();
        }
        return instance;
    }
}

LazyInitClass的竞争条件会破坏其正确性。假如两个线程A和B同时执行getInstance(),A看到instance为null,执行初始化,此时B也在检查instance是否为null,而instance是否为null,依赖于时序,是无法预期的,如果B检查也为null,则线程B也会执行初始化,得到两个不同的对象,和惰性初始化只初始化一次是矛盾的。

使用线程安全对象管理类的全部状态,可以维护类的线程安全性

将UnsafeCountServlet改造一下,代码如下:

public class SafeCountServlet{
    private final AtomicLong count = new AtomicLong(0);
    
    public long getCount(){return count.get();}
    
    public void service(ServletRequest req, ServletResponse res){
        BigInteger i = extractFromRequest(req);
        BigInteger[] factors = factor(i);
        count.incrementAndGet();
        encodeIntoResponse(res, factors);
    }
} 

java.util.concurrent.atomic包中包括了原子变量类,这些类用来实现数字和对象引用的原子状态转换,把long换成AtomicLong,可以确保所有访问计数器状态的操作都是原子的,计数器是线程安全的了,而计数器的状态就是SafeCountServlet的状态,所以SafeCountServlet也变成了线程安全的。

使用锁可以维护类的线程安全性

使用线程安全对象管理类的全部状态,可以维护类的线程安全性,但如果类中存在多个实例域,即有多个状态,仅仅加入更多的线程安全的状态变量时不够的。
为了保护状态的一致性,要在单一的原子操作中更新相互关联的的状态变量。
java提供了锁可以保护类的线程安全性。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,456评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,370评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,337评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,583评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,596评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,572评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,936评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,595评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,850评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,601评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,685评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,371评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,951评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,934评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,167评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,636评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,411评论 2 342

推荐阅读更多精彩内容

  • 从三月份找实习到现在,面了一些公司,挂了不少,但最终还是拿到小米、百度、阿里、京东、新浪、CVTE、乐视家的研发岗...
    时芥蓝阅读 42,169评论 11 349
  • 简书 賈小強转载请注明原创出处,谢谢! 线程安全的定义常常让人迷惑,搜索引擎会发现无数定义,比如: 多个线程同时执...
    賈小強阅读 221评论 0 0
  • 下面是我自己收集整理的Java线程相关的面试题,可以用它来好好准备面试。 参考文档:-《Java核心技术 卷一》-...
    阿呆变Geek阅读 14,735评论 14 507
  • 一.线程安全性 线程安全是建立在对于对象状态访问操作进行管理,特别是对共享的与可变的状态的访问 解释下上面的话: ...
    黄大大吃不胖阅读 822评论 0 3
  • 我听见 空调水落在窗台 树叶飘在草地 地球在转 灯光洒在被子上 夜 还很长…
    青鹅阅读 137评论 0 0