多线程与高并发- Synchronized锁

news2025/2/27 20:58:18

简介

synchronized 是 Java 语言的一个关键字,它允许多个线程同时访问共享的资源,以避免多线程编程中的竞争条件和死锁问题。synchronized可以用来给对象或者方法进行加锁,当对某个对象或者代码块加锁时,同时就只能有一个线程去执行。这种就是互斥关系,被加锁的区域称为临界区,而里面的资源就是临界资源。当一个线程进入临界区的时候,另一个线程就必须等待。
在这里插入图片描述
synchronized可以限制对某个资源的访问,但是它锁的并不是资源本身,可以锁住某个对象,只有线程拿到这把锁之后才能够去访问临界资源。如下代码,在我们想执行对count变量进行操作的时候,线程需要拿到o这个对象。

public class T1_Synchronized01 {
    private int count = 1;
    private Object o = new Object();

    public void m1() {
        synchronized (o) { // 必须先拿到o这个锁
            count++;
            System.out.println(Thread.currentThread().getName() + " count = " + count);
        }
    }
}

synchronized基础用法

1、通过对象进行锁

在代码里,可以通过创建一个对象,这样要想拿到临界资源,就必须先获得到这个对象的锁。

public class T1_Synchronized01 {
    private int count = 1;
    private Object o = new Object();

    public void m1() {
        synchronized (o) { // 必须先拿到o这个锁
            count++;
            System.out.println(Thread.currentThread().getName() + " count = " + count);
        }
    }
}
2、通过this

使用this代表锁住的是当前对象,这种方法等同直接把synchronized关键字加在方法前。

public class T1_Synchronized01 {
    private int count = 1;

    public void m2() {
        synchronized (this) { // 必须先拿到this的锁
            count++;
            System.out.println(Thread.currentThread().getName() + " count = " + count);
        }
    }

    public synchronized void m3() { // 与m2一样
        count++;
        System.out.println(Thread.currentThread().getName() + " count = " + count);
    }

}
3、锁定静态方法

锁定静态方法需要通过类.class,或者直接在静态方法上加上关键字。但是,类.class不能使用this来代替。注:在同一个类加载器中,class是单例的,这也就能保证synchronized能够只让一个线程访问临界资源。

public class T1_Synchronized01 {
    public static void m4() { // 静态方法
        synchronized (T1_Synchronized01.class) {
            System.out.println(Thread.currentThread().getName());
        }
    }

    public synchronized static void m5() {
       System.out.println(Thread.currentThread().getName());
    }
}
4、实验测试

①、首先测试一下,同步和非同步是否可以相互调用
定义两个线程,一个执行同步方法,一个执行非同步方法,如果不能够互相调用,那么,非同步线程就需要等待同步线程执行完之后在继续执行。

public class T3_Synchronized03 {
    public synchronized void one() {
        System.out.println(Thread.currentThread().getName() + " start one method");
        try {
            Thread.sleep(2000);
        } catch (InterruptedException e) {
            throw new RuntimeException(e);
        }
        System.out.println(Thread.currentThread().getName() + " end one method");
    }

    public void two() {
        System.out.println(Thread.currentThread().getName() + " execute two method");
    }

    public static void main(String[] args) {
        T3_Synchronized03 t = new T3_Synchronized03();
        new Thread(t::one, "第一个线程").start();
        new Thread(t::two, "第二个线程").start();
    }
}

从运行的结果可以看出是可以的。
在这里插入图片描述
②、读写不全加锁会怎样
通过购票与查询票数来进行模拟读写加锁问题。
首先,看以下代码是给读写都进行加锁了,在扣掉票数的时候,休眠了2秒,当线程执行了购票之后,通过多个线程去查询票数,每次启动线程会睡眠0.5秒。

public class T4_Synchronized {
    private int ticket = 100; // 模拟100张票

    public synchronized int getTicket() { // 读
        return this.ticket;
    }

    public synchronized void buy(int number) { // 写
        try {
            Thread.sleep(2000);
        } catch (InterruptedException e) {
            throw new RuntimeException(e);
        }
        ticket = ticket - number;
    }

    public static void main(String[] args) throws InterruptedException {
        T4_Synchronized bus = new T4_Synchronized();
        System.out.println("刚开始有票数:" + bus.getTicket());
        new Thread(() -> bus.buy(1)).start();
        for (int i = 1; i <= 10; i++) {
            Thread.sleep(500);
            int finalI = i;
            new Thread(() -> System.out.println("第" + finalI + "次查询余票数:" + bus.getTicket())).start();
        }

    }
}

运行之后,我们可以发现,数据是正确的,尽管是在查询的时候并没有睡眠0.5秒,显示数据依然是期望数据。
在这里插入图片描述
然而,当我们把读的锁去掉,运行代码,会发现,数据读出来了脏数据,为了更好的显示,查询票数的睡眠不要去掉。
在这里插入图片描述
③、synchronized的可重入性
定义一个类,类中有两个同步方法,他们锁的对象都是当前类,如果不能够重入,在one线程启动的时候就会死锁。在同步方法one中去调用同步方法two,当线程启动的时候,已经获取了对象的锁,等调用two方法的时候,同样是拿到了这个对象的锁。所以synchronized是可重入的。

public class T5_Synchronized {

    synchronized void one() {
        try {
            Thread.sleep(1000);
        } catch (InterruptedException e) {
            throw new RuntimeException(e);
        }
        two();
        System.out.println("one - thread-" + Thread.currentThread().getName() + " end");
    }

    synchronized void two() {
        try {
            Thread.sleep(2000);
        } catch (InterruptedException e) {
            throw new RuntimeException(e);
        }
        System.out.println("two - thread-" + Thread.currentThread().getName() + " end");
    }

    public static void main(String[] args) {
        T5_Synchronized t5 = new T5_Synchronized();
        new Thread(t5::one, "one1").start();
        new Thread(t5::one, "one2").start();
        new Thread(t5::one, "one3").start();
    }
}

实验结果
在这里插入图片描述
④、异常会释放锁
当线程执行过程中出现了异常,synchronized的锁会被释放,这样其他需要访问这个临界资源的线程就能进入执行。

public class T6_Synchronized {
    int count = 0;
    synchronized void add() {
        System.out.println("线程 " + Thread.currentThread().getName() + " start");
        while (true) {
            count++;
            System.out.println("线程 " + Thread.currentThread().getName() + " now count = " + count);
            if (count == 3) {
                throw new NullPointerException("人为异常");
            }
            if (count == 10) {
                throw new NullPointerException("测试结束");
            }
        }
    }

    public static void main(String[] args) throws InterruptedException {
        T6_Synchronized t = new T6_Synchronized();
        new Thread(t::add, "1").start();
        Thread.sleep(1000);
        new Thread(t::add, "2").start();
    }
}

当第一次异常抛出时,线程2就立即进入执行。
在这里插入图片描述

synchronized锁的底层原理

synchronized实现锁的基础就是Java对象头,synchronized锁会将线程ID存入mark word(对象头由标记字)。关于mark word,先简要了解一下Java对象。
在Hotspot 虚拟机中,对象在内存中的存储布局,可以分为三个区域:对象头(Header)、实例数据(Instance Data)、对齐填充(Padding)。synchronized主要是跟对象头有关系,在对象头中包含了标记字(mark word)、类指针(klass word)和 数组长度(array length)。也就是通过mark word的字节位数来表示各种锁状态。
在这里插入图片描述
synchronized锁在线程第一次访问的时候,实际上是没有加锁的,只是在mark word中记录了线程ID,这种就是偏向锁,默认是认为不会有多个线程抢着用,mark word是通过64bit来表示的,通过最低2位也就是锁标志位,偏向锁与无锁的值是01,轻量级锁用00表示,重量级锁用10表示,标记了GC的用11表示,无锁与偏向锁低2位是一致的,在倒数第3位有1位来表示偏向锁位:值为1表示偏向锁。
在这里插入图片描述

这里引用一张掘金博客上的图:https://juejin.cn/post/6978882583492821023

synchronized锁升级

● synchronized锁在线程第一次访问的时候,实际上是没有加锁的,只是在mark word中记录了线程ID,默认也就是使用偏向锁。
● 当第二个线程来争用的时候,此时第二个线程会占用cpu,循环等待锁的释放,这时候偏向锁也就升级为自旋锁。
● 当自旋10次之后,就会升级为重量级锁,重量级锁是不占用cpu,他是使用OS的。
当线程数较少、运行时间较短的时候是比较适合使用自旋锁,反之则比较适合重量级锁。


---------------------
作者:一个有梦有戏的人
来源:CSDN
原文:https://blog.csdn.net/qq_43843951/article/details/129107202
版权声明:本文为作者原创文章,转载请附上博文链接!
内容解析By:CSDN,CNBLOG博客文章一键转载插件

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

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

相关文章

Spring Cache常见问题解决

目录 一 报错:Null key returned for cache operation 二 报错&#xff1a;类型转换异常 三 取出的数据为null 一 报错:Null key returned for cache operation 这里报错有两种情况&#xff1a; 第一&#xff0c;如果你在新增的方法上使用Cacheable注解&#xff0c;那么肯定是…

chat使用

1.问题&#xff0c;Youve hit your usage limit. Please try again later. 2024年6月22号&#xff0c;提示达到使用限制次数。 一直用免费的&#xff0c;第一次遇见这个提示。 据说月初会重置。 感觉这个月也没有用多少次&#xff0c;怎么就达到限制了。 还有就是&#…

godot所有2D节点介绍

五十个2D节点介绍 2D节点介绍 前言一、Node2D二、sprite2D三、AnimatedSprite2D四、Camera2D五、PhysicsBody2D六、 RigidBody2D七、CharacterBody2D八、StaticBody2D九、joint2D十、DampedSpringJoint2D十一、GrooveJoint2D十二、PinJoint2D十三、Area2D十四、AnimatableBody2…

day3-xss漏洞(米斯特web渗透测试)

day3-xss漏洞&#xff08;米斯特web渗透测试&#xff09; XSSXss种类三种反射型1.反射型xss2.存储型xss3.DOM型xss XSS Xss有一部分是前端的有一部分不是前端的&#xff0c;我们来看一下&#xff0c;昨天的HTML注入修复方法应灵活使用。 HTML注入是注入一段HTML&#xff0c;那…

android studio 模拟器文件查找

android studio 模拟器文件查找 使用安卓模拟器下载文件后通常无法在系统硬盘上找到下载的文件&#xff0c;安卓 studio studio 其实提供了文件浏览工具&#xff0c;找到后可以直接使用 Android studio 打开 打开 Android studioview 菜单view > Tool Windows > Device…

三阶段复习

6.21 静态库与动态库 库有两种&#xff1a;静态库&#xff08;.a、.lib&#xff09;和动态库&#xff08;.so、.dll&#xff09;。所谓静态、动态是指链接。静态库在链接期把整个库文件都拷贝到可执行文件中&#xff0c;而动态库在链接期只是把索引文件拷贝到可执行文件中&…

数据结构~~时间、空间复杂度

目录 一、什么是数据结构 什么是算法 算法的复杂度 二、时间复杂度 三、空间复杂度 四、总结 一、什么是数据结构 数据结构(Data Structure)是计算机存储、组织数据的方式&#xff0c;指相互之间存在一种或多种特定关系的 数据元素的集合。 数据结构关注的是数据的逻辑结…

爬虫笔记14——爬取网页数据写入MongoDB数据库,以爱奇艺为例

下载MongoDB数据库 首先&#xff0c;需要下载MongoDB数据库&#xff0c;下载的话比较简单&#xff0c;直接去官网找到想要的版本下载即可&#xff0c;具体安装过程可以看这里。 pycharm下载pymongo库 pip install pymongo然后在在python程序中我们可以这样连接MongoDB数据库…

继电器十大品牌供应商

继电器是常用的元器件之一&#xff0c;如下是优秀供应商。 继电器品牌-中间继电器品牌-安全继电器品牌-固态继电器哪个品牌比较好-Maigoo品牌榜

【html】用html+css模拟Windows右击菜单

效果图&#xff1a; 在这个示例中&#xff0c;我为每个.second-list添加了一个.sub-menu的<div>&#xff0c;它包含了子菜单项。当鼠标悬停在.second-list上时&#xff0c;.sub-menu会显示出来。你可以根据需要调整这个示例以适应你的具体需求。 记住&#xff0c;这只是…

「动态规划」如何解决单词拆分问题?

139. 单词拆分https://leetcode.cn/problems/word-break/description/ 给你一个字符串s和一个字符串列表wordDict作为字典。如果可以利用字典中出现的一个或多个单词拼接出s则返回true。注意&#xff1a;不要求字典中出现的单词全部都使用&#xff0c;并且字典中的单词可以重复…

conda下安装32位版本python

前言&#xff1a;当前主流的系统为64bit系统&#xff0c;conda软件为64bit软件&#xff0c;因此使用conda创建虚拟环境安装python时默认安装的python为64bit版本&#xff0c;但部分研发场景需要调用32bit依赖&#xff0c;只能使用32bit的python&#xff0c;因此需要安装32bit的…

36.远程注入到入口点注入

免责声明&#xff1a;内容仅供学习参考&#xff0c;请合法利用知识&#xff0c;禁止进行违法犯罪活动&#xff01; 如果看不懂、不知道现在做的什么&#xff0c;那就跟着做完看效果&#xff0c;代码看不懂是正常的&#xff0c;只要会抄就行&#xff0c;抄着抄着就能懂了 上一…

ViT:5 Knowledge Distillation

实时了解业内动态&#xff0c;论文是最好的桥梁&#xff0c;专栏精选论文重点解读热点论文&#xff0c;围绕着行业实践和工程量产。若在某个环节出现卡点&#xff0c;可以回到大模型必备腔调或者LLM背后的基础模型重新阅读。而最新科技&#xff08;Mamba,xLSTM,KAN&#xff09;…

操作系统实验四:openEuler安装(openEuler配置静态网络、编写C或C++)

目录 一、实验要求 二、具体任务安排 1.安装openEuler &#xff08;1&#xff09;下载openEuler镜像 &#xff08;2&#xff09;使用vmware安装openEuler 2.在openEuler中编写C或者C测试程序 &#xff08;1&#xff09;安装g环境 &#xff08;2&#xff09;开始程序编码…

121.网络游戏逆向分析与漏洞攻防-邮件系统数据分析-邮件读取与发送界面设计

免责声明&#xff1a;内容仅供学习参考&#xff0c;请合法利用知识&#xff0c;禁止进行违法犯罪活动&#xff01; 如果看不懂、不知道现在做的什么&#xff0c;那就跟着做完看效果 现在的代码都是依据数据包来写的&#xff0c;如果看不懂代码&#xff0c;就说明没看懂数据包…

IEEE RAL 具有高运动性能的仿旗鱼机器人协同运动机制研究

水下机器人作为军用侦察、监测及攻击装置备受关注&#xff0c;目前传统水下机器人普遍采用螺旋桨作为推进器&#xff0c;但高噪音、高能耗等问题限制了应用范围。鱼类通过自然选择进化出优异的运动性能&#xff0c;特别是在海洋中游动速度快、机动性强的旗鱼。为了探究快速和高…

湖北民族大学2024年成人高等继续教育招生简章

湖北民族大学&#xff0c;这所承载着深厚文化底蕴和卓越教育理念的学府&#xff0c;在崭新的2024年再次敞开怀抱&#xff0c;热烈欢迎有志于深化学习、提升自我的成人学员们。今年的成人高等继续教育招生&#xff0c;不仅是学校对于终身教育理念的具体实践&#xff0c;更是为广…

【雷丰阳-谷粒商城 】【分布式高级篇-微服务架构篇】【13】压力压测JMeter-性能监控jvisualvm

持续学习&持续更新中… 守破离 【雷丰阳-谷粒商城 】【分布式高级篇-微服务架构篇】【13】压力压测JMeter-性能监控jvisualvm 压力测试概述性能指标 JMeter基本使用添加线程组添加 HTTP 请求添加监听器启动压测&查看分析结果JMeter Address Already in use 错误解决 性…

广东信息工程职业学院2024年成人高等继续教育招生简章

一、学校简介 广东信息工程职业学院位于广东省肇庆市&#xff0c;是一所具有一定办学规模&#xff0c;办学定位和培养目标明确&#xff0c;办学特色和追求鲜明&#xff0c;可持续发展的全日制普通高等学校&#xff0c;学院坚持以人为本&#xff0c;以德育人&#xff0c;以良好…