如何保证线程安全,看这里

上篇文章我们简单聊了什么是多线程,我想大家对多线程已经有了一个初步的了解,没看的没有放下文章链接 什么是线程安全,你真的了解吗?

上篇我们搞清楚了什么样的线程是安全的,我们今天先来看段代码:

public void threadMethod(int j) {

 int i = 1;

j = j + i;

}

大家觉得这段代码是线程安全的吗?

毫无疑问,它绝对是线程安全的,我们来分析一下为什么它是线程安全的?

我们可以看到这段代码是没有任何状态的,什么意思,就是说我们这段代码不包含任何的作用域,也没有去引用其他类中的域进行引用,它所执行的作用范围与执行结果只存在它这条线程的局部变量中,并且只能由正在执行的线程进行访问。当前线程的访问不会对另一个访问同一个方法的线程造成任何的影响。

两个线程同时访问这个方法,因为没有共享的数据,所以他们之间的行为并不会影响其他线程的操作和结果,所以说无状态的对象也是线程安全的。

                                    添加一个状态呢?

如果我们给这段代码添加一个状态,添加一个count,来记录这个方法并命中的次数,每请求一次count+1,那么这个时候这个线程还是安全的吗?

public class ThreadDemo {

int count = 0; // 记录方法的命中次数

public void threadMethod(int j) {

count++ ;

int i = 1;

j = j + i;

}

}

很明显已经不是了,单线程运行起来确实是没有任何问题的,但是当出现多条线程并发访问这个方法的时候,问题就出现了,我们先来分析下count+1这个操作。

进入这个方法之后首先要读取count的值,然后修改count的值,最后才把这把值赋值给count,总共包含了三步过程:“读取”一>“修改”一>“赋值”,既然这个过程是分步的,那么我们先来看下面这张图,看看你能不能看出问题:

如何确保线程安全?

可以发现,count的值并不是正确的结果,当线程A读取到count的值,但是还没有进行修改的时候,线程B已经进来了,然后线程B读取到的还是count为1的值,正因为如此所以我们的count值已经出现了偏差,那么这样的程序放在我们的代码中是存在很多的隐患的。

既然存在线程安全的问题,那么肯定得想办法解决这个问题,怎么解决?我们说说常见的几种方式。

1、synchronized

synchronized关键字就是用来控制线程同步的,保证我们的线程在多线程环境下,不被多个线程同时执行,确保我们数据的完整性,使用方法一般是加在方法上。

public class ThreadDemo {

int count = 0; // 记录方法的命中次数

public synchronized void threadMethod(int j) {

count++ ;

int i = 1;

j = j + i;

}

}

这样就可以确保我们的线程同步了,同时这里需要注意一个大家平时忽略的问题,首先synchronized锁的是括号里的对象,而不是代码,其次,对于非静态的synchronized方法,锁的是对象本身也就是this

当synchronized锁住一个对象之后,别的线程如果想要获取锁对象,那么就必须等这个线程执行完释放锁对象之后才可以,否则一直处于等待状态。

注意点:虽然加synchronized关键字可以让我们的线程变的安全,但是我们在用的时候也要注意缩小synchronized的使用范围,如果随意使用时很影响程序的性能,别的对象想拿到锁,结果你没用锁还一直把锁占用,这样就应了一句话:占着茅坑不拉屎,属实有点浪费资源。

2、Lock

先来说说它跟synchronized有什么区别吧,Lock是在Java1.6被引入进来的,Lock的引入让锁有了可操作性,什么意思?就是我们在需要的时候去手动的获取锁和释放锁,甚至我们还可以中断获取以及超时获取的同步特性,但是从使用上说Lock明显没有synchronized使用起来方便快捷。

我们先来看下一般是如何使用的:

private Lock lock = new ReentrantLock(); // ReentrantLock是Lock的子类

private void method(Thread thread){

lock.lock(); // 获取锁对象

try {

System.out.println("线程名:"+thread.getName() + "获得了锁");

// Thread.sleep(2000);

}catch(Exception e){

e.printStackTrace();

} finally {

System.out.println("线程名:"+thread.getName() + "释放了锁");

lock.unlock(); // 释放锁对象

}

}

进入方法我们首先要获取到锁,然后去执行我们业务代码,这里跟synchronized不同的是,Lock获取的所对象需要我们亲自去进行释放,为了防止我们代码出现异常,所以我们的释放锁操作放在finally中,因为finally中的代码无论如何都是会执行的。

写个主方法,开启两个线程测试一下我们的程序是否正常:

public static void main(String[] args) {

LockTest lockTest = new LockTest();

// 线程1

Thread t1 = new Thread(new Runnable() {

@Override

public void run() {

// Thread.currentThread()  返回当前线程的引用

lockTest.method(Thread.currentThread());

}

}, "t1");

// 线程2

Thread t2 = new Thread(new Runnable() {

@Override

public void run() {

lockTest.method(Thread.currentThread());

}

}, "t2");

t1.start();

t2.start();

}

结果:

可以看出我们的执行是没有任何问题的。

其实在Lock还有几种获取锁的方式,我们这里再说一种就是tryLock()这个方法跟Lock()是有区别的,Lock在获取锁的时候如果拿不到锁就一直处于等待状态,直到拿到锁,但是tryLock()却不是这样的,tryLock是有一个Boolean的返回值的,如果没有拿到锁直接返回false,停止等待,它不会像Lock()那样去一直等待获取锁。

我们来看下代码:

private void method(Thread thread){

// lock.lock(); // 获取锁对象

if (lock.tryLock()) {

try {

System.out.println("线程名:"+thread.getName() + "获得了锁");

// Thread.sleep(2000);

}catch(Exception e){

e.printStackTrace();

} finally {

System.out.println("线程名:"+thread.getName() + "释放了锁");

lock.unlock(); // 释放锁对象

}

}

}

结果:我们继续使用刚才的两个线程进行测试可以发现,在线程t1获取到锁之后,线程t2立马进来,然后发现锁已经被占用,那么这个时候它也不在继续等待。

似乎这种方法感觉不是很完美,如果我第一个线程拿到锁的时间比第二个线程进来的时间还要长,是不是也拿不到锁对象,那我能不能用一中方式来控制一下,让后面等待的线程可以需要等待5秒,如果5秒之后还获取不到锁,那么就停止等,其实tryLock()是可以进行设置等待的相应时间的。

private void method(Thread thread) throws InterruptedException {

// lock.lock(); // 获取锁对象

// 如果2秒内获取不到锁对象,那就不再等待

if (lock.tryLock(2,TimeUnit.SECONDS)) {

try {

System.out.println("线程名:"+thread.getName() + "获得了锁");

// 这里睡眠3秒

Thread.sleep(3000);

}catch(Exception e){

e.printStackTrace();

} finally {

System.out.println("线程名:"+thread.getName() + "释放了锁");

lock.unlock(); // 释放锁对象

}

}

}

结果:看上面的代码我们可以发现,虽然我们获取锁对象的时候可以等待2秒,但是我们线程t1在获取锁对象之后执行任务缺花费了3秒,那么这个时候线程t2是不在等待的。

我们再来改一下这个等待时间,改为5秒,再来看下结果:

private void method(Thread thread) throws InterruptedException {

       // lock.lock(); // 获取锁对象

       // 如果5秒内获取不到锁对象,那就不再等待

       if (lock.tryLock(5,TimeUnit.SECONDS)) {

           try {

               System.out.println("线程名:"+thread.getName() + "获得了锁");

           }catch(Exception e){

               e.printStackTrace();

} finally {

               System.out.println("线程名:"+thread.getName() + "释放了锁");

               lock.unlock(); // 释放锁对象

           }

       }

   }

结果:这个时候我们可以看到,线程t2等到5秒获取到了锁对象,执行了任务代码。

这就是使用Lock来保证我们线程安全的方式,其实Lock还有好多的方法来操作我们的锁对象,这里我们就不多说了,大家有兴趣可以看一下API。

PS:现在你能做到如何确保一个方法是线程安全的吗?

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

推荐阅读更多精彩内容

  • 1.理清文章各章节脉络和逻辑; 2.带走文章的亮点; 3.含大量自己的感想、心得; 4.与读过的书、已有的知识建立...
    王新印阅读 1,919评论 0 0
  • 爸妈在家吃好的用好的,女儿们在外面省吃俭用拼死拼活挣来的死工资都被父母每次的“装可怜”压榨的干干净净! ...
    辣妈不怕辣阅读 699评论 12 7
  • 日渐扭曲变形的社会 逐渐消磨殆尽的人性 圣洁无暇的天使成为传说 平静不起波澜的现世里 那些钢筋铜铸的丛林里 一个一...
    赵金祥阅读 506评论 0 2