Java多线程锁

news2025/1/15 1:14:53

多线程锁

本专栏学习内容又是来自尚硅谷周阳老师的视频

有兴趣的小伙伴可以点击视频地址观看

Synchronized

Synchronized是Java中锁的一种实现方法,我们需要了解他锁在什么地方,锁的类型有哪些

阿里巴巴开发手册规定:

高并发时,同步调用应该去考量锁的性能消耗,能用无锁数据结构,就不要用锁;能锁区块,就不要锁整个方法体;能用对象锁,就不要用类锁

说明:尽可能使加锁的代码块工作量尽可能的小,避免在锁代码块中调用RPC方法

同步方法

操控两个线程、一个资源类

资源类

class Phone{
    public synchronized void sendEmail(){
        try {TimeUnit.MILLISECONDS.sleep(1000);} catch (InterruptedException e) {throw new RuntimeException(e);}
        System.out.println("----sendEmail");
    }

    public synchronized void sendSMS(){
        System.out.println("----sendSMS");
    }

    public void hello(){
        System.out.println("hello");
    }
}

一个资源对象执行两个同步方法

线程A执行sendEmail()时会加锁,锁的对象是new Phone()也就是堆空间中的那个对象,在线程B调用sendSMS()时,锁对象也是new Phone(),所以需要等待线程A执行完毕才能获取锁

public class SyncDemo1 {
    public static void main(String[] args) throws InterruptedException {
        Phone phone = new Phone();
        new Thread(() -> {
            phone.sendEmail();
        },"a").start();
        //保证线程a先运行
        TimeUnit.MILLISECONDS.sleep(200);
        new Thread(() -> {
            phone.sendSMS();
        },"b").start();
    }
}

//结果
----sendEmail
----sendSMS

一个资源对象执行一个同步方法和一个普通方法

这个就比较简单,因为hello()不需要获取锁,可以直接执行

public static void main(String[] args) throws InterruptedException {
    Phone phone = new Phone();
    new Thread(() -> {
        phone.sendEmail();
    },"a").start();
    //保证线程a先运行
    TimeUnit.MILLISECONDS.sleep(200);
    new Thread(() -> {
        phone.hello();
    },"b").start();
}

//结果
hello
----sendEmail

两个资源对象执行两个同步方法

因为这两个方法的锁对象不同,所以互不影响

public static void main(String[] args) throws InterruptedException {
    Phone phone = new Phone();
    Phone phone2 = new Phone();
    new Thread(() -> {
        phone.sendEmail();
    },"a").start();
    //保证线程a先运行
    TimeUnit.MILLISECONDS.sleep(200);
    new Thread(() -> {
        phone2.sendSMS();
    },"b").start();
}

//结果
----sendSMS
----sendEmail

静态同步方法

资源类

class Phone{
    public static synchronized void sendEmail(){
        try {TimeUnit.MILLISECONDS.sleep(1000);} catch (InterruptedException e) {throw new RuntimeException(e);}
        System.out.println("----sendEmail");
    }

    public static synchronized void sendSMS(){
        System.out.println("----sendSMS");
    }

    public void hello(){
        System.out.println("hello");
    }
}

一个资源对象执行两个静态同步方法

对于静态同步方法,锁住的是Phone这个Class对象,也就是存在与方法区中的Phone,所以不管是一个资源对象还是多个资源对象,调用静态同步方法,使用的都是同一个锁

public static void main(String[] args) throws InterruptedException {
    Phone phone = new Phone();
    new Thread(() -> {
        phone.sendEmail();
    },"a").start();
    //保证线程a先运行
    TimeUnit.MILLISECONDS.sleep(200);
    new Thread(() -> {
        phone.sendSMS();
    },"b").start();
}

//结果
----sendEmail
----sendSMS

同步代码块

同步代码块的锁,就是括号中填的对象,可以是对象锁,也可以是类锁

synchronized (this) {
    
}

字节码角度分析

使用javap -c xxxx.class可以反编译字节码文件,如果要看详细信息可以使用javap -v xxxx.class

同步代码块

public class SyncDemo2 {
    Object object = new Object();

    public void m1() {
        synchronized (object) {
            System.out.println("m1 method");
        }
    }
    public static void main(String[] args) throws InterruptedException {

    }
}

在JVM中是由monitore来控制锁的,但是在同步代码块中,发现有一个获取锁,有两个释放锁

第二个释放锁有点保护机制的意思,如果同步代码块中出现异常,无法正常释放锁,会有异常的释放方式

image-20230725162440971

同步方法和静态同步方法

    public synchronized void m2() {
        System.out.println("m2 method");
    }
    
    public static synchronized void m3() {
        System.out.println("m2 method");
    }

JVM中使用ACC_SYNCHRONIZED来表示当前方法是同步方法,使用ACC_STATIC来表示该方法为静态方法

image-20230725162921402

公平锁、非公平锁

非公平锁

非公平锁是一种线程同步机制,它允许新的线程在获取锁时,不考虑其他等待线程的顺序,有可能插队获取到锁资源。相对于公平锁来说,非公平锁在一定程度上可以提高系统的吞吐量,但可能导致某些线程长时间地等待。

模拟卖票案例

一共50张票,交给a、b、c三个窗口去卖

public class LockDemo1 {
    public static void main(String[] args) {
        Ticket ticket = new Ticket();
        new Thread(() -> {for(int i = 0;i < 60;i++) ticket.buy();},"a").start();
        new Thread(() -> {for(int i = 0;i < 60;i++) ticket.buy();},"b").start();
        new Thread(() -> {for(int i = 0;i < 60;i++) ticket.buy();},"c").start();
    }
}

class Ticket {
    private int sum = 50;
	//默认使用非公平锁
    private ReentrantLock lock = new ReentrantLock();

    public void buy() {
        try {
            lock.lock();
            if (sum > 0) {
                System.out.println(Thread.currentThread().getName() + "卖出第  " + sum + " 张票,还剩 " + --sum);
            }
        } finally {
            lock.unlock();
        }
    }
}

通过观察结果可以发现,可能有一个窗口把50张票卖完,也有可能一个窗口一张票都卖不出

image-20230726140159232

公平锁

公平锁是一种线程同步机制,它按照线程请求锁的顺序来分配锁资源,保证线程获取锁的顺序与其请求锁的顺序一致。公平锁可以避免线程饥饿的情况,但可能降低系统的吞吐量。

模拟买票案例

可以使用new ReentrantLock(true)来创建公平锁,可以从结果看出,运行一段时间后会保证顺序获取锁

image-20230726140425728

如何选择

一般来说,对于线程执行顺序要求不高的,完全可以使用非公平锁,因为线程之间的切换是非常消耗时间的,非公平锁可以提高吞吐量。

可重入锁

简单理解为:可以重复进入的同步锁,当然是有前提条件的

概念

可重入锁是一种线程同步机制,也称为递归锁。它允许同一个线程在拥有锁的情况下多次进入被锁定的代码块,而不会造成死锁。可重入锁在保证线程安全的同时,提供了更大的灵活性和方便性。

代码演示

synchronizedReentrantLock都属于可重入锁

synchronized

如果不是可重入锁,按照同步锁的理论知识,外层获取object锁时,第二层应该就不能获取到该锁,程序应该会卡死在那里,但是我们发现程序正常的执行完毕,由此可见synchronized是可重入锁

public static void main(String[] args) throws InterruptedException {
    final Object object = new Object();
    synchronized (object) {
        System.out.println(Thread.currentThread().getName() + "进入外层");
        synchronized (object) {
            System.out.println(Thread.currentThread().getName() + "进入中层");
            synchronized (object) {
                System.out.println(Thread.currentThread().getName() + "进入内层");
            }
        }
    }
}

//结果
main进入外层
main进入中层
main进入内层

ReentrantLock

ReentrantLock锁对象是ReentrantLock类的实例,同样也是可重入锁

public static void main(String[] args) {
    ReentrantLock lock = new ReentrantLock();
    lock.lock();
    try {
        System.out.println(Thread.currentThread().getName() + "进入外层");
        lock.lock();
        try {
            System.out.println(Thread.currentThread().getName() + "进入中层");
            lock.lock();
            try {
                System.out.println(Thread.currentThread().getName() + "进入内层");
            }finally {
                lock.unlock();
            }
        }finally {
            lock.unlock();
        }
    }finally {
        lock.unlock();
    }
}

//结果
main进入外层
main进入中层
main进入内层

原理

每一个锁对象拥有一个锁计数器和一个指向持有该锁的线程的指针

当执行monitorenter时,如果锁对象的计数器为0,那么说明他没有被其他线程所持有,JVM会将锁对象的持有线程设置为当前线程,并且将其计数器+1。

在目标锁的计数器不为0的情况下,如果锁对象的持有线程是当前线程,那么JVM可以将其计数器+1,否则需要等待。

当执行monitorexit时,JVM会将对象的计数器-1,计数器为0代表锁已经被释放。

死锁

死锁是多线程编程中一种常见的情况,指的是两个或多个线程无限期地等待对方释放资源,从而导致程序无法继续执行的状态。

public class SycnDemo04 {
    static Object a = new Object();
    static Object b = new Object();

    public static void main(String[] args) {
        new Thread(()->{
            synchronized (a){
                System.out.println(Thread.currentThread().getName() + "获取了锁a");
                try {
                    TimeUnit.SECONDS.sleep(1);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                synchronized (b){
                    System.out.println(Thread.currentThread().getName() + "获取了锁b");
                }
            }
        },"A").start();

        new Thread(()->{
            synchronized (b){
                System.out.println(Thread.currentThread().getName() + "获取了锁b");
                try {
                    TimeUnit.SECONDS.sleep(1);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                synchronized (a){
                    System.out.println(Thread.currentThread().getName() + "获取了锁a");
                }
            }
        },"B").start();
    }
}

//结果
A获取了锁a
B获取了锁b

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

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

相关文章

大语言模型LLM技术赋能软件项目管理和质量保障︱微软中国高级研发经理步绍鹏

微软中国高级研发经理步绍鹏先生受邀为由PMO评论主办的2023第十二届中国PMO大会演讲嘉宾&#xff0c;演讲议题&#xff1a;大语言模型LLM技术赋能软件项目管理和质量保障。大会将于8月12-13日在北京举办&#xff0c;敬请关注&#xff01; 议题内容简要&#xff1a; 本次分享将…

809 协议相关

809 协议 目录概述需求&#xff1a; 设计思路实现思路分析1.概念2.业务流程3.详细过程4.相关过程 参考资料 Survive by day and develop by night. talk for import biz , show your perfect code,full busy&#xff0c;skip hardness,make a better result,wait for change,ch…

Windows上安装Docker Desktop

运行环境 Windows 10Docker Desktop 4.21.1 安装步骤 步骤1&#xff1a; 勾掉"Use WSL 2 instead of Hyper-V(recommended)"&#xff08;原因见小插曲2章节&#xff09; 步骤2&#xff1a; 安装完成 步骤3&#xff1a; 运行Docker Desktop 步骤4&#xff1a; …

面试了无数家公司整理的软件测试面试题【含答案】

1、自动化代码中,用到了哪些设计模式? 单例设计模式 工厂模式PO设计模式数据驱动模式面向接口编程设计模式 2、什么是断言( Assert) ? 断言Assert用于在代码中验证实际结果是不是符合预期结果&#xff0c;如果测试用例执行失败会抛出异常并提供断言日志 3、什么是web自动化测…

STM32 串口实验(学习一)

本章将实现如下功能&#xff1a;STM32通过串口和上位机对话&#xff0c;STM32在收到上位机发过来的字符串后&#xff0c;原原本本返回给上位机。 STM32 串口简介 串口作为MCU的重要外部接口&#xff0c;同时也是软件开发重要的调试手段&#xff0c;其重要性不言而喻。现在基本…

centos7设置网桥网卡

安装bridge-utils yum install bridge-utils修改ens33 网卡 TYPEEthernet BOOTPROTOnone DEFROUTEyes IPV4_FAILURE_FATALno IPV6INITyes IPV6_AUTOCONFyes IPV6_DEFROUTEyes IPV6_FAILURE_FATALno NAMEens33 UUID04b97484-25c8-45c7-8c8c-e335e8080e10 DEVICEens33 ONBOOTye…

TEE GP(Global Platform)认证方案

TEE之GP(Global Platform)认证汇总 一、GP认证方案 二、GP认证方案分类 参考&#xff1a; GlobalPlatform Certification - GlobalPlatform

【读书笔记】《生命册》- 李佩甫 - 2012年出版

不停的阅读&#xff0c;然后形成自己的知识体系。 2023.07. 读 《生命册》是河南作家李佩甫于2012年发表的长篇小说&#xff0c;作家出版社发行&#xff0c;全书共12章约38万字&#xff0c;与之前发表的《羊的门》、《城的灯》组成“平原三部曲”&#xff0c;2015年获得第九届…

恶意软件已渗透商业环境,超40万个企业凭证被窃取

网络安全公司 Flare 与 BleepingComputer 分享的一份报告显示&#xff0c;在对暗网和 Telegram 渠道上出售的近 2000 万条泄密数据日志进行分析后&#xff0c;发现信息窃取恶意软件已实现了对商业环境的严重渗透。 信息窃取程序是一种恶意软件&#xff0c;可窃取存储在 Web 浏…

Jmeter 测试 JMS (Java Message Service)/ActiveMQ 性能

目录 前言 ActiveMQ 介绍 准备工作 编写jndi.properties添加到ApacheJMeter.jar 中 下载 ActiveMQ 配置 Jmeter 进行测试 点对点 (Queues 队列) 配置 Jmeter 进行测试 发布/订阅 (Topic 队列) 配置发布 Publisher 配置订阅 Subscriber 总结 前言 在进行性能测试时&a…

Java API文档工具:小白如何游刃有余编写Java中的API使用手册?

Java API文档工具&#xff1a;小白如何游刃有余编写Java中的API使用手册&#xff1f; 对于初学者来说&#xff0c;Java API文档是解决问题和学习语言的重要工具。本文将介绍Java API文档工具的使用方法&#xff0c;帮助小白们轻松编写出优秀的API使用手册&#xff0c;供日后的…

PaaS低代码平台源码:预置3000+应用集成插件,强大的建模引擎,快速构建自己的应用程序和流程

低代码PaaS平台是一款基于 Salesforce Platform 的开源替代方案&#xff0c;支持多种企业应用场景&#xff0c;包括但不限于CRM、ERP、OA、BI、IoT、大数据等。无论是传统企业还是新兴企业&#xff0c;都可以使用管理后台快速构建自己的应用程序和流程。 技术架构&#xff1a;微…

【软件测试】单元测试工具---Junit详解

1.junit 1.1 junit是什么 JUnit是一个Java语言的单元测试框架。 虽然我们已经学习了selenium测试框架&#xff0c;但是有的时候测试用例很多&#xff0c;我们需要一个测试工具来管理这些测试用例&#xff0c;Junit就是一个很好的管理工具&#xff0c;简单来说Junit是一个针对…

emacs打开git仓库下多个子工程的根目录问题解决案

emacs打开git仓库下多个子工程的根目录问题解决案 问题描述 如题所述&#xff0c;这个问题困扰我很久了&#xff0c;一直没搜到完整的解决方案。这次终于乘着空闲时间&#xff0c;研究了projectile.el源码找到了方案。 问题场景具体描述下: 我自己有一个私人git仓库&#x…

CISA学习笔记-第一章、信息系统审计过程

传统的审计三方关系理论指明&#xff0c;审计作为独立于会计记录之外的一项重要职能&#xff0c;是公司财务信息公允可靠的有力保障&#xff0c;制约着会计行为&#xff0c;制衡了会计权力。 1. IS审计和保障标准、指南、工具 职业道德规范 信息技术保证框架&#xff08;ITAF&a…

Log4j源码解析

Log4j源码解析 主要流程 Logger logger Logger.getLogger(Main.class); 1、通过Logger.getLogger(Class clazz) 或 Logger.getLogger(String name)进入。 2、加载LogManager进jvm, 执行静态代码块执行初始化, 创建出RepositorySelector实例及LoggerRepository实例(Hierarchy…

13. linux指令

文章目录 一些linux的常用指令一、linux常用命令二、查看系统是32还是64位三、修改字符集四、Linux的时间问题1.查看和修改Linux的时区2.查看和修改Linux的时间 五、解压六、yum源配置1.yum介绍2.查看有无 yum安装包 &#xff0c;有的话卸载&#xff1a; 七、防火墙操作&#x…

达梦HSEM管理平台部署测试

目录 前期准备... 3 一、HSEM外部接入部署... 6 1、元数据库准备... 6 2、创建DmhsWebService. 6 3、启动DMHSWeb服务... 6 4、创建代理服务... 7 5、运行代理服务... 8 6、运行 Tomcat 服务... 8 7、登录HSEM管理平台... 8 二、Agent代理接入... 10 1、修改配置文件…

【VUE】npm打包报错 Syntax Error: Error: Cannot find module ‘imagemin-gifsicle‘

一. Syntax Error: Error: Cannot find module ‘imagemin-gifsicle’ npm run build 报错&#xff0c;报错如下 原因 这个错误消息显示缺少了 imagemin-gifsicle 模块&#xff0c;而它是 image-webpack-loader 的依赖项&#xff0c;导致构建失败。解决 &#xff08;1&#xf…

多值提取至点(样地因子提取)

1.导入因子tif&#xff0c;和样地表 2.值提取至点 输入对应的数据 也可以采用多值提取至点。 可以选择多个tif影像 提取后会将对应的字段添加在表后面 打开属性表可以看到 采用转换工具表转EXCEL可以导出为表格