• 博客园logo
  • 会员
  • 众包
  • 新闻
  • 博问
  • 闪存
  • 赞助商
  • HarmonyOS
  • Chat2DB
    • 搜索
      所有博客
    • 搜索
      当前博客
  • 写随笔 我的博客 短消息 简洁模式
    用户头像
    我的博客 我的园子 账号设置 会员中心 简洁模式 ... 退出登录
    注册 登录
赫连小伍
博客园    首页    新随笔    联系   管理    订阅  订阅

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. 将方法定义为静态方法,静态方法属于某个类,不属于具体的实例
 

-- 赫连小伍原创,转载请注明出处,微信公众号同名
 

posted @ 2020-09-15 17:53  赫连小伍  阅读(211)  评论(0)    收藏  举报
刷新页面返回顶部
博客园  ©  2004-2025
浙公网安备 33010602011771号 浙ICP备2021040463号-3