浅析Java线程安全

概述

线程安全是多线程编程时的计算机程序代码中的一个概念。在拥有共享数据的多条线程并行执行的程序中,线程安全的代码会通过同步机制保证各个线程都可以正常且正确的执行,不会出现数据污染等意外情况。

什么是线程安全?

多个线程访问同一个对象时,如果不用考虑这些线程在运行环境下的调度和交替执行,也不需要进行额外的同步,或者在调用方进行任何其他操作,调用这个对象的行为都可以获得正确的结果,那么这个对象就是线程安全的。或者说,一个类或者程序所提供的接口对于线程来说是原子操作或者多个线程之间的切换不会导致该接口的执行结果存在二义性,也就是说我们不用考虑同步的问题。

线程安全问题的起因

线程安全问题都是由全局变量及静态变量引起的。若每个线程中对全局变量、静态变量只有读操作,而无写操作,一般来说,这个全局变量是线程安全的;若有多个线程同时,执行写操作,一般都要考虑线程同步,否则就有可能影响线程安全。

图1

如图1所示,这段代码在单线程或者串行的多线程情况下显然是线程安全的。如果在并行的多线程情况下,图1中的方法就会出现线程安全问题。笔者尝试开启了3条线程,并行访问图1中所示的方法,每个线程循环10次,得到了图2所示的结果:

图2

可以看到,这里出现了两个26,然而根据代码逻辑,在运行正确的情况下根本不可能会得出这种结果,出现这种问题的根源就是线程安全问题。考虑以下情况:假设A线程在进入方法后,取得了变量count的值,并且还未能改变变量count的值,与此同时,B线程也进入方法,同样取得了变量count的值。此时,A线程与B线程取得的变量count的值完全相同,因此A线程和B线程执行方法体之后将会得出同样的结果,正如图2所示的出现两个26的情况。

如何保证线程安全?

当多个线程访问同一个方法时,无论这些如何交替执行,调用方都不需要做任何同步工作,其运行所得仍然是正确的结果,那么这个方法就是线程安全的。在图3所示的代码中,方法threadMethod()中并未出现全局变量或静态变量,毫无疑问,该方法是一个绝对线程安全的方法。

图3

若在图3的方法中加入一个全局变量,如图4所示。

图4

则此方法显然会出现线程安全问题,虽然单线程运行起来并不会出现任何问题,但是多线程并发访问此方法,全局变量count将会导致线程安全问题,如图5所示。

图5

A线程和B线程进入这个方法后首先读取变量count的值,接着分别执行count++,修改变量count的值,此过程可视为分三步进行,即“读取->修改->赋值”。从图5中可以发现,变量count的值并非正确的结果。当A线程读取到count的值且并未修改时,B线程也读取到了count的值,此时count的值仍为1,正因如此,变量count的计算结果将会出现偏差。针对上述情况,Java中一般有两种方法来保证线程安全。

1.synchronized

关键字synchronized用来控制线程同步,确保多线程环境下,加上此关键字的方法不会被并行执行,如图6所示。

图6

需要注意的是,关键字synchronized将会对非静态方法所属的对象进行加锁。当synchronized对一个对象进行加锁之后,其他线程必须等待正在访问此对象的线程执行完毕,释放锁之后才可以进行访问。

2.Lock

Lock是在Java1.6之后被引入进来,Lock与synchronized相比,前者的引入让锁有了更强的可操作性,即在实际使用过程中可以手动获取锁和释放锁,甚至还拥有中断获取以及超时获取的同步特性,但从使用便捷角度来看,Lock明显没有synchronized方便快捷,如图7所示。

图7

进入方法首先需要获取锁对象,接着继续执行业务代码。与synchronized不同的是,Lock获取的锁对象需要手动释放,为了应对代码出现异常的情况,通常在finnally中执行释放锁的操作,因为finally的特性使得释放锁的操作必定会执行。

结语

在大型应用系统中,多线程的使用不可避免,此时线程安全问题就成为了亟需解决的问题,不考虑线程安全问题将会给整个系统埋下严重的安全隐患,只有在保证线程安全的前提下,应用系统才能正确稳定地运行。

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

推荐阅读更多精彩内容

  • 本文是我自己在秋招复习时的读书笔记,整理的知识点,也是为了防止忘记,尊重劳动成果,转载注明出处哦!如果你也喜欢,那...
    波波波先森阅读 11,223评论 4 56
  • 进程和线程 进程 所有运行中的任务通常对应一个进程,当一个程序进入内存运行时,即变成一个进程.进程是处于运行过程中...
    胜浩_ae28阅读 5,080评论 0 23
  • 1.解决信号量丢失和假唤醒 public class MyWaitNotify3{ MonitorObject m...
    Q罗阅读 862评论 0 1
  • 引言 此文是iOS中的多线程编程:重温GCD(一)系列的第二部分。将会辅以一定的例子简单讲解一些更深层次的API使...
    ZhengLi阅读 335评论 0 3
  • 这是我测试的
    Tsrign阅读 344评论 0 0