synchronized简单理解

news2025/1/15 17:09:03

一、简述

1.1 synchronized介绍

synchronized是一种互斥锁,也成为同步锁,它的作用是保证在同一时刻,被修饰的代码块或方法只会有一个线程执行,以到达保证并发安全效果。在JDK1.6以前,很多人称之为重量级锁,性能不高。但是在JDK1.6以后,对sychronized进行了一些优化,引入了偏向锁,轻量级锁,以及重量级锁。这个时候,synchrionized会根据线程的竞争程度对锁进行升级和降级。

二、使用

2.1 synchronized使用重要性

容易引发线程安全

什么是线程安全?

当多个线程同事,对一个共享资源进行非原子操作(如:修改某个共享资源得数据时),将会出现线程安全问题。

 比如下面代码,加了synchronized和没有加synchronized有明显的区别

public class ThreadTestByPool {

    //获取CPU个数
    private static int cpuCount = Runtime.getRuntime().availableProcessors();
    private static ThreadPoolExecutor threadPool = new ThreadPoolExecutor(cpuCount, cpuCount * 2, 20,
            TimeUnit.SECONDS, new LinkedBlockingQueue<>(20000));
    static {
        System.out.println("我的cpu颗数: " + cpuCount);
    }
    //成员变量,可被多个线程 共享
    private static long n = 0L;

    public static void main(String[] args) throws InterruptedException {
        CountDownLatch countDownLatch = new CountDownLatch(10000);
        for (int i = 0; i < 10000; i++) {
            threadPool.execute(() -> {
                try {
                    //synchronized (ThreadTestByPool.class) {
                        n++;
                    //}
                } finally {
                    countDownLatch.countDown();
                }
            });
        }
        //等待所有线程执行完毕以后再打印
        countDownLatch.await();
        System.out.println("n++的结果集为:" + n);
    }

}

如果没有加synchronized,三次打印效果

第一次:

我的cpu颗数: 12
n++的结果集为:9988

第二次:

我的cpu颗数: 12
n++的结果集为:9987

第三次:

我的cpu颗数: 12
n++的结果集为:9981

每次执行的结果集都会不一样,但是执行的结果都是不正确的

去掉上面synchronized注释,加了synchronized以后执行,无论执行了几次,返回的n值都是正确的

返回的结果集:

我的cpu颗数: 12
n++的结果集为:10000

2.2 synchronized使用

2.2.1 synchronized三种使用方式

  • 修饰实例方法:作用相当于给实例加锁
  • 修饰静态方法:作用相当于给对象加锁
  • 修饰代码块:指定加锁对象
public class ThreadTest {

    //给方法加锁
    public synchronized  void test(){

    }
    //给静态方法加锁
    public static synchronized  void staticTest(){

    }

    public void test2(){
        // 代码块
        synchronized (this){

        }
        synchronized (ThreadTest.class){

        }
    }

}

注意点:

  • synchronized关键字不能被继承
  • 定义接口方法不能使用synchronized
  • 构造方法不能使用synchronized

2.2.2 案例

  ThreadPoolUtil线程池工具地址:线程池工具类_java-zh的博客-CSDN博客

public class SyncTest {
    //获取CPU个数
    private static int cpuCount = Runtime.getRuntime().availableProcessors();

    static {
        System.out.println("我的cpu颗数: " + cpuCount);
    }

    //成员变量,可被多个线程 共享
    private static  long a = 0L;
    private static long b = 0L;
    private static long c = 0L;

    public static void main(String[] args) throws InterruptedException {
        CountDownLatch countDownLatch = new CountDownLatch(10000);
        SyncTest lockobj = new SyncTest();
        for (int i = 0; i < 10000; i++) {
            ThreadPoolUtils.execute(() -> {
                try {
                    lockobj.sync(countDownLatch);
                    sync2(countDownLatch);
                    lockobj.sync3(countDownLatch);
                }finally {
                    countDownLatch.countDown();
                }
            });
        }
        // 等待所有的线程执行完再打印
        countDownLatch.await();
        System.out.println("a=" + a);
        System.out.println("b=" + b);
        System.out.println("c=" + c);
    }

    /**
     * 修饰生源方法,其使用的锁对应时当前类所在的实例对象(或者说当前方法的调用对象),也就是this
     *
     * @param countDownLatch
     */
    private synchronized void sync(CountDownLatch countDownLatch) {
        try {
            a++;
        } finally {
           // countDownLatch.countDown();
        }
    }

    /**
     * 修饰静态方法,使用的锁对象是当前类
     *
     * @param countDownLatch
     */
    private synchronized static void sync2(CountDownLatch countDownLatch) {
        try {
            b++;
        }  finally {
            //countDownLatch.countDown();
        }
    }

    private void sync3(CountDownLatch countDownLatch) {
        try {
            synchronized (countDownLatch) {
                c++;
            }
        }  finally {
            //countDownLatch.countDown();
        }
    }

}

 结果:不管执行几次,得出的结果都是正确的

我的cpu颗数: 12
a=10000
b=10000
c=10000

三、 synchronized锁住的真正资源 

参考文献:万字长文分析synchroized_wx6402d9406dac7的技术博客_51CTO博客

1、被synchronized修饰的成员方法,在编译的时候,该方法会生成一个'ACC_SYNCHRONIZED'标记

2、被synchronized修饰的静态方法,在编译的时候,该方法也会生成一个'ACC_SYNCHRONIZED'标记

3、被synchronized修饰的代码块,在执行代码块的逻辑之前,会执行'monitorenter'汇编指令获取锁,执行完逻辑以后,紧接着是'monitorexit'指令,此时,锁被当前持有的线程释放。

3.1 真正的资源ObjectMonitor

实际上,不管被标记的ACC_SYNCHRONIZED方法还是被插入的monitorenter/monitorexit指令的同步块,最终JVM底层都是一个逻辑。在进入该方法或同步块时,必须竞争到给定对象对应的ObjectMonitor上的资源(这里的竞争具体表现为:某个线程通过CAS操作来给定对象对应的ObjectMonitor上的owner指针指向自己,CAS操作成功,就代表获取锁成功,CAS失败,就代表未获取到锁)。同时,必须清楚一点,每个对象都有一个ObjectMonitor与之相对应,且唯一。

        每个对象都和一个监听器相关联。当且仅当监听器于某个对象关联时,它才会被锁定。执行monitorenter的线程获得与对象关联的监听器所有权。

  • 如果与对象关联的监听器条目(count值)计数为0,则线程进入监听器并将其条目设置为1。g该线程就是监听器的所有者。
  • 如果线程已拥有对象关联的监听器,它会重新进入监听器,增加其count计数值
  • 如果另一个线程已拥有与锁对象关联的监听器,线程将会阻塞,知道监听器的count为0,然后再次尝试获得所有权

总结:抢锁的操作对应到JVM底层来说,其实就是CAS设置onwer指针指向当前抢锁线程操作,设置成功即抢锁成功,设置失败即抢锁失败。

3.2 锁对象与ObjectMonitor的关系

  1. 执行monitorenter的线程试图获得与指定对象(synchronized块中的锁对象)关联的ObjectMonitor,每个对象都有一个监听器(ObjectMonitor)与之相关联
  2. 当且仅当监听器(ObjectMonitor)和某个对象产生关联时,ObjectMonitor标志已被锁定/持有

如图(下面的图为重量锁):

 解释:

  1. 尝试获取:当前程获取锁时,会进行cas操作,尝试将owner指针指向当前线程,如果获取成功,则进入同步块执行逻辑
  2. 获取失败后:从cxq队首插入,包装了当前线程的node
  3. 当持有锁的线程释放后:首先肯定的就是他将会owner置为null,好让出资源,然后会从EntryList(如果没有从cxq)队列中挑选一个线程抢锁,被选中的线程叫做Heir presumptive,即叫做"假定继承人","假定继承人"尝试获取锁,但synchronized是非公平的,所以"假定继承人"也不一定能获取锁(所以这也是叫"假定继承人"的原因)。
  4. 当持有锁的线程调用Object的wait方法后:则会将当前线程加入到WaitSet中
  5. 当被Object.notify/notifyAll方法唤醒以后:会将对应的线程从WaitSet移动到cxq或EntryList中去。需要注意的是,当调用一个锁对象的wait或notify/notifyAll方法时,如果当前锁的状态是偏向锁或轻量级锁则会先膨胀成重量级锁。

四、锁的升级

参考文献:并发编程笔记:synchronized关键字浅析_zhoutaoping1992的博客-CSDN博客

4.1 锁的状态

无锁 -> 偏向锁 -> 轻量级锁 -> 重量级锁

  • 无锁:在没有开启偏向锁或者偏向锁延迟未到或者批量撤销后,对象创建完成后,没有任何线程使用该对象来加锁,此时该对象状态是无锁。对应的Java对象头lock=1,biased_lock=0。(biased_lock=0的含义表示不可偏向)
  • 偏向锁:对象创建完成后处于可偏向状态,某一个对象使用该对象来加锁,那么这个对象就偏向这个线程。对应的java头lock=1,biased_lock=1。
  • 轻量级锁:当有多个线程交替使用某个对象来加锁,并无无竞争的情况下,该对象会成为轻量级锁。对应的Java对象头lock=00
  • 重量级锁:当有多个线程竞争使用某个对象来加锁,该对象会成为重量级锁。对应的JAV对象头lock=10

4.2 JVM相关参数

        UseBiasedLocking是否使用偏向锁,默认true
BiasedLockingStartupDelay偏向锁开启延迟时间,默认4000ms
BiasedLockingBulkRebiasThreshold偏向锁重偏向阈值,默认20次
BiasedLockingBulkRevokeThreshold偏向锁撤销阈值,默认40次

4.3 synchronized关键字执行流程

如图(不考虑重偏向和批量撤销的情况):

  1. 无锁(lock为01,biased_lock为0,不可偏向状态)状态没有任何路径升级为偏向锁状态
  2. 轻量锁和重量锁在synchronized代码块执行完毕后会将锁对象重置为无锁状态,而偏向锁执行完毕后不会重置锁状态
  3. 升级路径只有无锁或偏向锁升级为轻量级锁和轻量级锁升级为重量级锁
  4. 在偏向锁升级为轻量级锁时,会进行锁撤销操作,将锁专题修改为无锁状态,然后再升级。 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/620268.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

作为一名仓库管理人员,如何有效地管理仓库?

作为一名仓库管理人员&#xff0c;如何有效地管理仓库&#xff1f; 有效仓库管理主要可以分为四个方面&#xff1a; 出入库 库存调拨 库存盘点 虚拟库存/实际库存管理 当然仓库管理最基本的硬件条件还是需要准备好的&#xff0c;比如将仓库分一下区域&#xff0c;以便之后商…

layui(3)——内置模块弹出层

弹出层 1.基础参数 使用模块layer layui.use(layer, function(){var layer layui.layer;}); <script>layui.use(layer, function () {var layer layui.layer;layer.open({// layer提供了5种层类型。可传入的值有&#xff1a;0&#xff08;信息框&…

极简操作!跟着官方教程,下载并使用汉化版Figma!

随着云端协作设计工具的发展&#xff0c;Figma 作为一款基于云端的国外云端协作工具&#xff0c;尽管无需下载即可在线使用&#xff0c;但在国内却常常遇到加载缓慢、需要刷新重进的问题。此外&#xff0c;Figma 在实际使用过程中&#xff0c;常常需要搭配其他软件或第三方插件…

vue3+element plus,使用分页total修改成中文

vue3element plus&#xff0c;使用分页total修改成中文 使用element plus的分页功能 el-pagination 的时候&#xff0c;total属性显示是英文 这是我建的一个新项目&#xff0c;总数显示的Total 1000 我们的需求是显示中文&#xff0c;共 1000 条 这个就很尴尬&#xff0c;组件…

JAVA基础,区别于C++

JAVA 基础 1. 数据类型 字节型 byte 1 字节 -128~127 短整型 short 2 字节 -32768~32767 整型 int 4 字节 -2147483648~2147483647 长整型 long 8 字节 -9223372036854775808L~92…

初学Nginx要掌握哪些概念?

文章目录 为什么要使用Nginx&#xff1f;什么是Nginx&#xff1f;Nginx的作用&#xff1f;反向代理负载均衡动静分离 为什么要使用Nginx&#xff1f; 小公司项目刚刚上线的时候&#xff0c;并发量小&#xff0c;用户使用的少&#xff0c;所以在低并发的情况下&#xff0c;一个…

iNotes(WebMail)中的附件显示

大家好&#xff0c;才是真的好。 1995年Domino R4/4.5中加入了HTTP功能&#xff0c;从此便支持从用户从Web浏览器上查看邮件.一开始称之为Domino Web Access&#xff0c;后来更名为IBM Lotus iNotes&#xff0c;简称iNotes&#xff0c; 对于不熟悉Notes/Domino的人来说&#…

达梦数据库dblink测试(DM-ORACLE19c及DM8-DM8)

目录 DM与DM创建DBLINK. 3 一、配置203服务器... 3 1、主服务器203开启mal服务... 3 2、主服务器203配置dmmal.ini文件... 3 3、重启203数据库... 4 二、配置200服务器... 4 三、创建dblink. 4 DM与Oracle数据库创建dblink. 5 一、DM服务器&#xff1a;... 6 1、初始…

祝愿莘莘学子高考顺利!Good luck on your Gaokao!

《登科后》 唐孟郊 昔日龌龊不足夸&#xff0c;今朝放荡思无涯。 春风得意马蹄疾&#xff0c;一日看尽长安花。 Gone are all my past woes! What more have I to say? My body and my mind enjoy their fill today. Successful, faster runs my horse in vernal breeze;…

ArrayList 的底层原理和源码分析

tip&#xff1a;作为程序员一定学习编程之道&#xff0c;一定要对代码的编写有追求&#xff0c;不能实现就完事了。我们应该让自己写的代码更加优雅&#xff0c;即使这会费时费力。 推荐&#xff1a;体系化学习Java&#xff08;Java面试专题&#xff09; 文章目录 一、简介二、…

Cookie Session

第一章 会话技术 1.1 什么是会话 web会话可简单理解为&#xff1a;用户开一个浏览器&#xff0c;访问某一个web网站&#xff0c;在这个网站点击多个超链接&#xff0c;访问服务器多个web资源&#xff0c;然后关闭浏览器&#xff0c;整个过程称之为一个会话. 它是指浏览器和服…

URL到页面: 探索网页加载的神秘过程

当我们从浏览器的地址栏输入 URL, 按下回车, 再到最后出现需要的网页界面, 这中间究竟发生了什么, 接下来就一步步进行解析. 主要是如下过程: 输入网址DNS 解析客户端发送 HTPP 请求建立 TCP 连接服务器处理请求, 计算响应, 返回响应浏览器渲染页面关闭连接 本篇中只是概述整…

AUTOSAR-OS的调度机制-调度表(没理解透,继续更新)

什么是调度表&#xff1a; 1. 调度表由一系列按时间先后顺序排序的终结点组成&#xff0c;其中每个终结点都有自己的任务&#xff0c;有的终结点可能是激活一系列的任务&#xff0c;有的是设置一系列的事件&#xff0c;还有的可能是既激活一系列的任务又设置一系列的事件。 调…

数据结构之庖丁解牛八大排序算法,附动图说明过程(上)

目录 一、排序的概念以及应用 二、常见排序算法的实现 1.插入排序 1.1直接插入排序 b.实现直接插入排序 1.2希尔排序&#xff08;缩小增量排序&#xff09; 2.选择排序 2.1直接选择排序 2.2堆排序 3.交换排序 3.1冒泡排序 一、排序的概念以及应用 1.排序的概念 所谓排序&#x…

Mysql数据库--修改root密码的几种方法(忘记密码知道密码)

Mysql数据库--修改root密码的几种方法&#xff08;忘记密码&知道密码&#xff09; &#x1f53b;一、知道密码情况--修改root密码⛳ 1.1 方式1&#xff1a;alter 命令修改⛳ 1.2 方式2&#xff1a;set password命令修改 &#x1f53b;二、忘记密码情况-修改root密码⛳ 2.1 …

华为OD机试真题 Java 实现【找车位】【2023 B卷 100分】,附详细解题思路

一、题目描述 停车场有一横排车位&#xff0c;0代表没有停车&#xff0c;1代表有车。至少停了一辆车在车位上&#xff0c;也至少有一个空位没有停车。 为了防剐蹭&#xff0c;需为停车人找到一个车位&#xff0c;使得距停车人的车最近的车辆的距离是最大的&#xff0c;返回此…

打造高质量视频,创造视觉奇观!Camtasia 2023为你升级!

嘿&#xff0c;伙计&#xff01; 在这个全新版本中&#xff0c;我们迎来了焕然一新的动画控制和更简化的特效制作流程&#xff0c;让创作变得更高效。 不仅如此&#xff0c;全新的背景去除和动画光标功能也让视频拥有全新的视觉体验。让我们先谈谈光标&#xff0c;这个细节或…

计算机视觉:风格迁移

风格迁移 本节将介绍如何使用卷积神经网络&#xff0c;自动将一个图像中的风格应用在另一图像之上&#xff0c;即风格迁移&#xff08;style transfer&#xff09; (Gatys et al., 2016)。 这里我们需要两张输入图像&#xff1a;一张是内容图像&#xff0c;另一张是风格图像。…

Mac使用DBeaver连接达梦数据库

Mac使用DBeaver连接达梦数据库 下载达梦驱动包 达梦数据库 在下载页面随便选择一个系统并下载下来。 下载下来的是zip的压缩包解压出来就是一个ISO文件&#xff0c;然后我们打开ISO文件进入目录&#xff1a;/dameng/source/drivers/jdbc 进入目录后找到这几个驱动包&#x…

Vue 2和Vue 3路由Router创建的区别简记(在main.js文件中引入的区别和router的js文件中创建语法的区别)

Vue 2和Vue 3路由Router创建的区别即Router3.0和Router4.0的创建区别简记 1、版本的搭配&#xff1a; Vue 2到Vue 3的改版升级&#xff0c;同样的带来Vue Router的升级。创建Vue项目之后&#xff0c;我们可以在package.json文件中看到&#xff0c;Vue 2创建的项目往往是与Vue…