Java中synchronized的同步原理

​本文介绍在java开发中,怎么利用synchronized保证代码的同步执行,避免跳坑,废话不多说,直接上demo。

package com.helianxiaowu.demo;/** * @title  synchronized同步demo * @desc  注意:为了演示方便,代码中多线程使用直接new的方式,这种方式不可取,会造成服务器资源消耗。 *              工作中如果用到多线程,尽量使用线程池维护线程 * @author  helianxiaowu * @date  2020/1/14 上午 11:12 */public class SynchronizedDemo {    public static void main(String[] args) {        Demo demo = new Demo();        // 启用两条线程        Thread t1 = new Thread(() -> demo.print("Thread1"));        Thread t2 = new Thread(() -> demo.print("Thread2"));        t1.start();        t2.start();    }    static class Demo {        public void print(String flag) {            System.out.println(flag + " start");            try {                Thread.sleep(3000);            } catch (InterruptedException e) {                e.printStackTrace();            }            System.out.println(flag + " end");        }    }}

以上代码的执行结果为

 

现在我们在print方法上加上synchronized关键字,代码如下

执行结果如下,从执行结果中我们可以看到代码变成同步执行了

 

原理

JVM为每一个类或实例维护了一个监视器monitor,当出现synchronized关键字时monitor会做出如下判断

1. 判断monitor是否已有所有者,如果没有,则允许线程进入,并把计数器从0变为1

2. 如果monitor已经拥有所有者,则当前线程进入等待状态,并把计数器加1

3. 当线程退出时,计数器减1。当计数器为0时,monitor失去所有者

 

原理搞清楚了,我们在进行深入的研究。接着对代码进行修改,调用print方法分别使用不同的实例。修改完的代码如下

代码执行结果如下,从执行结果来看,synchronized关键字并没有起到作用,这是因为什么

 

注意事项

synchronized使用时必须是同一个实例,因为JVM为每个实例分配了一个monitor,不同的实例是不同的monitor在监听,所以实现不了同步。保证同一个实例的方法有很多种,比如:

1. 把java对象交给spring管理

2. 使用单例模式

3. 将方法定义为静态方法,静态方法属于某个类,不属于具体的实例

 


扫码关注我的公众号

 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值