《java并发编程实战》第二章:线程安全

第一章主要通过多线程如何重要,多线程将来要应用的越来越多,像是全书的一个引子,就是告诉读者,你选择本书没有错,这本书讲述的就是java中最重要的多线程部分,是程序开发技术中的屠龙刀。诚然本书(英文版本)是在2006年既java诞生10周年时候出版的。距离我2017年看本书已经过十年多的时间,虽然此书仍然是java并发界,乃至程序并发界一本经典书籍,此书中的绝大多数观点依然没有过时,但是计算机技术日新月异,十年前的很多观点到现在可能已经不在适用了,这需要在阅读过程中多加辨析。
第二章在第一章的基础上告诉读者在重要的多线程开发中线程安全的重要性,随着计算机硬件的发展,我们可以利用多核心cpu并发能力来让多个程序同时运行,这是这样计算机的效率就非常高,这是多线程开发的优点。但是要想享受这个优点,就要遵守多线程开发的“规则”。这个规则就是第二章介绍的-线程安全。作者在章首表达了两个观点:

A program that omits needed synchronization might appear to work, passing its tests and performing well for years, but it is still broken and may fail at any moment.
It is far easier to design a class to be thread-safe than to retrofit it for thread safety later.
《Java Concurrency in Practice》

两个观点分别强调线程安全在多线程环境的必要性:

  1. 你本应该应用并发的代码即使已经在生产环境中运行了很久,并且表现良好,但是它们也随时有可能崩溃。
  2. 去修复一个类或者一段代码来保证它们是线程安全的远比当初就把它们设计成线程安全的要复杂的多。

对于第一点,我是感触颇深的,在刚工作不久有个同事开发一个功能,需要用到多线程技术,同事也非常注意多线程开发过程中的线程安全问题,没日没夜的忙了快两个月,后来又联合程序,测试,产品等测试将近一个月。可是到了功能上线的时候还是出现了死锁等线程安全问题。此后每周改功能上线的时候都会或多或少的爆出线程安全问题,直到功能正式上线1个月后才算稳定。在严密的测试也不能排查所有线上可能发生的多线程问题。
第二点,可以参考修改老代码...改过别人代码的小伙伴们都懂。修复非线程安全类为线程安全的类和完善别人的代码本质上没有区别。都是抄剩饭。
说了半天线程安全,那么什么是线程安全呢?


黑人问号脸
黑人问号脸

看下本书作者给的定义:

A class is thread‐safe if it behaves correctly when accessed from multiple threads, regardless of the scheduling or interleaving of the execution of those threads by the runtime environment, and with no additional synchronization or other coordination on the part of the calling code.

一个类是线程安全的要满足两个条件:
1. 不管java的运行时(runtime environment)怎么安排线程访问此类的顺序都能得到最终你认为结果。
比如你和你和你老婆大人公用一张银行卡(当然是你的工资卡了)。你和你老婆大人都可以向这个银行卡里面存钱同时也可以取钱。如果你月薪500元,在你发工资后,银行就不能让你和你老婆大人分别取出500元,不管你先取钱然后你老婆大人再取钱,还是你老婆大人先取钱然后你再取钱,亦或是你们两个协调好了同时取钱。如果你和老婆大人手里一共有500元,那么卡里余额绝对是0元,如果不是0元,而是1000元,那么劝你赶快去银行“自首”,因为任何意外所得都归国有。这里银行卡就相当于一个线程安全的类,而你和你老婆大人就是两个不同的线程。
2. 线程安全的类的“安全性”是它自身属性,使用它的类不需要提供额外的安全机制。
这点就很好解释了,银行保证你卡里面的500元钱不管你怎么折腾都不会多,不管你是去ATM取钱,拿支付宝转账,用微信买瓶饮料。钱数该减少10元时候它绝对不会增加1元,该增加1元的时候它也绝对不会减少1分。

有童鞋可能会说,这两个条件不是天经地义的吗?本来就应该这样啊,但是啊,你认为的天经地义是习惯成自然的结果,就和你丢了钱去找警察,买个东西要交钱都是后天条件反射。对于咿呀学语的小孩纸来说,这些都不那么自然,都要问下为什么。以下面的银行卡类和ATM类,对刚才的两条进行详细说明。

public class BankCard {
    private int money;


    public BankCard(String userName,String userPassword){

        /**
         *  用 userName,userPassword查找数据库,初始化实例类BankCard。
         */
    }

    /**
     * 查看银行卡余额
     * @return
     */
    public int getMoney(){
        return money;
    }

    /**
     * 向银行卡中加钱
     * @param addNum 加钱数
     */
    public void addMoney(int addNum){
        money = money + addNum;
    }

    /**
     * 从银行卡中减钱
     * @param subNum  减钱数
     */
    public void subMoney(int subNum){
        if (money - subNum < 0) {
            return;
        }
    }
}
public class ATM {

    private BankCard card;


    public ATM(String userName,String userPassword) {
        this.card = new BankCard(userName, userPassword);
    }

    /**
     * 通过ATM向指定银行卡存钱
     * @param money
     */
    public void  saveMoney(int money){
        card.addMoney(money);
    }

    /**
     * 通过ATM从指定银行卡中取钱
     * @param money
     */
    public void drawMoney(int money){
        card.subMoney(money);
    }
} 

结合 《java并发编程实战》之java内存模型 和线程安全的类的定义,我们知道BankCard是线程不安全的,当你和你妻子在不同的ATM机上通过输入账号密码来取钱时候(不用插卡的ATM机,没见过吧)。用代码模拟如下:
你自己存钱:

ATM atm_you = ATM("10223","****");
atm_you.saveMoney(500); 

你老婆存钱:

ATM atm_wife = ATM("10223","*****");
atm_wife.saveMoney(1);

通过ATM的构造方法我们知道,当账号密码相同时候,会拿到同一张银行卡。如果你老婆和你同时存钱(假设开始卡里面没有一分钱),那么存钱结果可能是1,可能是500,当然也可能是你和银行都希望的501。为什么会这样呢?来看下面这张图片:


线程_内存模型图
线程_内存模型图

java 中的对象实例是存在 JMM中的堆内存的,类中的方法的局部变量存储规则如下:

  • 如果局部变量是基本类型(int,long,boolean等)其值是存在栈内存的,例如上图中的addMoney ,它是ATM的saveMoney方法的参数,其值存在栈内存。
  • 如果局部变量是非基本变量(各种类Integer,Long,或者本例中的ATM等),那么对这些类的实例的索引(atm_you atm_wife等)存在栈内存中,里面放着具体实例的堆内存地址。

所以对于“你自己存钱”这个线程和“你老婆存钱”这个线程你们每个人都自己独立操作空间,这就相当于线程的栈内存,然后两个线程分别把500元和1元放进ATM机器,这个500元和1元就是局部变量,分别存储在各自的栈内存中。但是两个线程必须要操作同一张银行卡,怎么办呢?银行卡放在银行仓库中,这就相当于堆内存。于是就分别创建一个专线链接这张银行卡,这就相当于索引,然后把其中的money值分别读过来,操作完(money+500、money+1)再写回堆内存。这“读”,“操作”,“写”在计算机中虽然时间都特别短,但是总还是需要时间的,比如“你自己存钱”和“你老婆存钱”线程同时在12整进行。但是你的ATM机器反映灵敏一些。就会很可能出现如下图问题:


线程时序图
线程时序图

两个线程虽然同时开始操作,但是因为各种原因,老婆的“读”慢了一些,你先把500这个结果存回银行卡,现在数字变成了500,3毫秒后,老婆线程的1元到账了,最后你银行卡中只有1元钱。为了便于理解,上图只假设“读”过程时间不同,其实“操作” “写”这个些过程的时间都不可能相同,线程的开始时间也不一定相同,那么结果就可以是1 500 501等不同情况了。因为不能总是达到我们的预期要求(银行卡存款额501) 所以BankCard这个类就是非线程安全的。
那么怎么才能让BankCard这个类线程安全呢? 答案是加锁,上述BankCard之所以不是线程安全的,根本原因是因为“你存钱线程”和“你老婆存钱线程”可以同时操作这唯一的一张银行卡,如果有一种机制能够在A线程想操作BankCard时候判断现在有没有其他线程正在操作BankCard,如果有,那么不许可A线程操作BankCard,如果没有那么A线程可以操作BankCard,java提供了锁机制来实现如上功能。来看BankCardSafe类:

public class BankCardSafe {
    private int money;


    public BankCardSafe(String userName,String userPassword){

        /**
         *  用 userName,userPassword查找数据库,初始化实例类BankCard
         */
    }

    /**
     * 查看银行卡余额
     * @return
     */
    public synchronized int getMoney(){
        return money;
    }

    /**
     * 向银行卡中加钱
     * @param addNum 加钱数
     */
    public synchronized  void addMoney(int addNum){
        money = money + addNum;
    }

    /**
     * 从银行卡中减钱
     * @param subNum  减钱数
     */
    public synchronized void subMoney(int subNum){
        if (money - subNum < 0) {
            return;
        }
    }
}

相比于BankCard类,BankCardSafe类只是在每个方法前面加个了synchronized关键字,这就相当于给方法加了一把锁,我们称这种锁为内置锁(intrinsic lock),这种锁在保护和它最近的一对{ }中的所有代码,当程序执行到 { 时自动加锁,当程序执行到 } 时,自动释放锁。锁在同一时间只能被一个线程拥有。如果线程没有锁,那么它就不能执行锁所保护的方法,然后线程就一直等待,直到别的线程释放了这个锁,它拿到这个锁后才能继续执行。当我们把ATM类的成员变量改成BankCardSafe时:

public class ATM {

    private BankCardSafe card;


    public ATM(String userName,String userPassword) {
        this.card = new BankCardSafe(userName, userPassword);
    }

    /**
     * 通过ATM向指定银行卡存钱
     * @param money
     */
    public void  saveMoney(int money){
        card.addMoney(money);
    }

    /**
     * 通过ATM从指定银行卡中取钱
     * @param money
     */
    public void drawMoney(int money){
        card.subMoney(money);
    }
}

这时程序的执行时序图就会变成下面的样子:


程序时序图
程序时序图

由于同一时间只能由一个线程抢到同一把锁,假设“老婆存钱线程”先拿到锁(老婆优先合情合理啊),那么只有它执行完“读” “操作” “写” 这些对BankCardSafe.money 变量的操作后,其释放完锁,"你存钱线程"才能拿到锁,然后才能继续执行你对BankCardSafe.money的“读” “操作” “写”等过程。这样就自然不会出错了。

至此,《Java Concurrency in Pracetice》 第二章的大体知识就介绍完了。剩下的还有重入锁,和一些关于执行效率的知识,感觉没有太大必要说,对于初学者来说。

参考文献:

1.《Java Concurrency in Pracetice》

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

推荐阅读更多精彩内容