专栏首页海纳周报synchronized关键字的语义

synchronized关键字的语义

上一篇文章,我们讲到,如果发生了多个线程共同访问一个全局变量的时候,就会发生各种意料之外的情况。其实现实生活中有很多这样的例子。我举一个例子。

一群人都要过河,但是河面上只有一只独木舟,除了船夫,一次只能带一个人。所有到达河边的人都想往船上抢,难免把船搞翻了。为了解决这个问题,我们可以在河边上设一个售票处,谁先抢到票,谁就可以上船,没有抢到票的,就只能等待下一次,船返回来,再去抢下一张票。

好了,在多线程编程中,我们也可以引入这样一个售票处,让线程先去抢票,抢到票的,就可以使用这只小船,抢不到的,就继续等待。这个售票处,就是 synchronized 了。

synchronized 方法

当一个方法加上synchronized 关键字以后,就只能让一个线程来执行这个方法了。只让一个线程的意思并不是只把这个方法指定给某个固定的线程,而是说一次只能有一个线程来调用这个函数。

我们把昨天的那个程序改一下,就很清楚了:

public class TestTwo {
    public int total = 0;

    public synchronized void incTotal() {
        total += 1;
    }

    public static void main(String[] args) throws Exception{
        TestTwo test = new TestTwo();

        Thread t1 = new Thread() {
            public void run() {
                for (int i = 0; i < 5_000; i++) {
                    test.incTotal();
                }
            }
        };

        Thread t2 = new Thread() {
            public void run() {
                for (int i = 0; i < 5_000; i++) {
                    test.incTotal();
                }
            }
        };

        t1.start();
        t2.start();

        t1.join();
        t2.join();

        System.out.println(test.total);
    }
}

现在不论你执行多少次,最后的结果一定会是10000了,这是因为我们把加1的操作用synchronized 保护起来了。一旦一个线程进入到了 incTotal 以后,其他的线程就不能再进入了。这样,我们就可以保证这个加法是完整而且独立的,其他的线程完全不能打扰到它了。

synchronized block

有时候,我们使用synchronized 修饰一个方法,会显得太重了一些。往往需要使用这种互斥进行保护的只是几个语句,而不是一个方法。那我们还可以使用语句块,它的语法是这样的:

synchronized(object) {
    // do something
}

例如,上面的例子,我们还可以这样写:

public class TestThree {
    public int total = 0;

    public static void main(String[] args) throws Exception{
        TestThree test = new TestThree();

        Thread t1 = new Thread() {
            public void run() {
                for (int i = 0; i < 5_000; i++) {
                    synchronized (test) {
                        test.total += 1;
                    }
                }
            }
        };

        Thread t2 = new Thread() {
            public void run() {
                for (int i = 0; i < 5_000; i++) {
                    synchronized (test) {
                        test.total += 1;
                    }
                }
            }
        };

        t1.start();
        t2.start();

        t1.join();
        t2.join();

        System.out.println(test.total);
    }
}

看,我们在这里就不用再定义一个synchronized方法了。而是在一个对象上加上这个互斥就可以了。

实际上,这里都不定要使用 test 这个对象。例如,我可以用一个完全不相干的对象来做互斥:

Object o = new Object();
synchronized(o) {
   do_something...
}

只要这个变量是两个线程都能访问就可以了。

本文分享自微信公众号 - HinusWeekly(gh_4b8b4eda4e40),作者:海纳

原文出处及转载信息见文内详细说明,如有侵权,请联系 yunjia_community@tencent.com 删除。

原始发表时间:2018-01-17

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • Java线程模型

    大家新年好。2018年的第一期来得晚了一些。因为年底有很多工作要做,加班多了一些,公众号停更了两周。 今天借着知乎上一个关于线程模型的问题,我正好可以讲一下Ja...

    海纳
  • 用Atomic实现锁

    一直想写ReentrantLock,就得先介绍AbstractQueueSynchronizer,可是我觉得这样写,不过瘾,我把代码贴一遍,懂的人自己就能找到这...

    海纳
  • 【第二期】一次学透java.io

    java.io是新手学习Java的第一个难点。因为这个package中的东西比较多,也比较复杂,另外加上一些接口太过于面向对象了,更加增大了学习的难度。这一期,...

    海纳
  • 用不用 synchronized 的区别?

    今天一起来认识认识 「synchronized」 这个一面试就会被提到的关键字。这一篇不会讲太多理论,主要先熟悉熟悉一下最简单的用法。只讨论一个问题:「方法没用...

    LieBrother
  • 分享一款JVM线程堆栈在线分析工具

    JVM大家可能都知道是个什么玩意-Java虚拟机,但是到底是个什么鬼?相信即使工作3-5年的程序员可能也不大了解。 ? 如题所述,今天与大家分享的是如何分析...

    小柒2012
  • Java中多线程的使用(超级超级详细)线程安全+线程锁原理解析+保证线程安全的三种方式 (同步代码块+同步方法+lock锁) 5

    当我们使用多线程访问同一个资源时,且多个线程对资源有写的 操作就容易出现线程安全问题,java为了解决线程安全问题引入了同步机制来解决,即在一个线程使用公共代码...

    一只胡说八道的猴子
  • 分享一款JVM线程堆栈在线分析工具

    JVM大家可能都知道是个什么玩意-Java虚拟机,但是到底是个什么鬼?相信即使工作3-5年的程序员可能也不大了解。

    小柒2012
  • 零基础学编程009:只显示2位小数

    我们仍要继续解决这个问题:如何用Python打印这篇枯燥的《复利数据表》? (1+0.01) ^ 1 = 1.01 (1+0.01) ^ 2 = 1.02 (...

    申龙斌
  • go 笔记

    结构体:定义的一组字段的集合 结构体初始化,new(Student) == &Student{}是等价的。 接口:定义的一组方法的集合

    槽痞
  • 一样的打游戏,不一样的酷

    夏乙 问耕 假装发自 凹非寺 量子位 出品 | 公众号 QbitAI ? 假期模式已经开启了~ 学生党已经赋闲在家,工业党不少也已带着橘子返乡。 阖家欢乐,面对...

    量子位

扫码关注云+社区

领取腾讯云代金券