【Java 基础】21 多线程同步与锁

news2024/11/26 9:56:43

文章目录

    • 1.存在的问题
    • 2.使用同步解决问题
      • 1) synchronized
      • 2) volatile
      • 3) 锁
    • 总结

用多线程过程中,有可能出现 多个线程同时处理(获取或修改等)同一个数据,这个时候就 会发生数据不同步的问题, 因此出现了同步和锁来保证多个线程可以安全的处理同一个数据。

1.存在的问题

例如:火车售票窗口售票,假如我们有 2 个窗口(相当于开启了 2 个线程),同时卖 10 张票
在这里插入图片描述

两个窗口的操作流程都如下:

1)购票者到窗口

2)窗口访问票匣子获取余票数量

3)票匣子返回余票数量

4)售票窗口判断,若存在票(大于 0 )则卖给他

那么,假如票匣子就剩下 1 张票啦!但是不巧的是两个窗口同时查看余票数量,都发现还有 1 张票,**又都卖给了购票者……**这就是多线程存在的数据不同步问题_

示例代码:

public class Demo {
    public static void main(String[] args) throws InterruptedException {
        ThreadDemo threadDemo = new ThreadDemo();
        new Thread(threadDemo,"售票窗口1").start();
        new Thread(threadDemo,"售票窗口2").start();
    }
}

class ThreadDemo implements Runnable {
    private int ticketCount = 10;
    @Override
    public void run() {
        String tName = Thread.currentThread().getName();
        while (true) {
            if (ticketCount <= 0) {
                return;
            }
            try {
                Thread.sleep(200);
                System.out.println(tName + "成功卖了一张票!余票:" + (ticketCount-- - 1));
            } catch (InterruptedException e) {
                throw new RuntimeException(e);
            }
        }
    }
}

输出结果:

售票窗口1成功卖了一张票!余票:9
售票窗口2成功卖了一张票!余票:8
售票窗口1成功卖了一张票!余票:7
售票窗口2成功卖了一张票!余票:6
售票窗口2成功卖了一张票!余票:5
售票窗口1成功卖了一张票!余票:4
售票窗口2成功卖了一张票!余票:3
售票窗口1成功卖了一张票!余票:2
售票窗口1成功卖了一张票!余票:1
售票窗口2成功卖了一张票!余票:0
售票窗口1成功卖了一张票!余票:-1

2.使用同步解决问题

同步(Synchronization) 是一种协调多个线程执行的机制,它能够确保在同一时刻只有一个线程访问共享资源。主要通过关键字 synchronizedvolatile 以及 锁对象 等手段来实现同步。

1) synchronized

关键字 synchronized 用于修饰方法或代码块,保证在同一时刻只有一个线程能够执行被 synchronized 修饰的代码。以下是两种使用方式:

  • 修饰方法
public synchronized void test() {
    // 同步的代码块
}
  • 修饰代码块
public void someMethod() {
    // 非同步的代码块

    synchronized (lockObject) {
        // 同步的代码块
    }

    // 非同步的代码块
}

2) volatile

关键字 volatile 用于声明变量,保证变量的可见性。被 volatile 修饰的变量对所有线程可见,当一个线程修改了这个变量的值,其他线程能够立即看到修改后的值。

public class Demo {
    private volatile int ticketCount = 10;
}

3) 锁

Java 提供了很多种锁,常用的有 synchronized 关键字、ReentrantLockRead/Write Lock 等 。

  • ReentrantLock

    它支持可重入锁,允许一个线程多次获取同一把锁。

    import java.util.concurrent.locks.Lock;
    import java.util.concurrent.locks.ReentrantLock;
    
    public class Demo {
        private final Lock lock = new ReentrantLock();
    
        public void test() {
            lock.lock();
            try {
                // 同步的代码块
            } finally {
                lock.unlock();
            }
        }
    }
    

    ReentrantLock 提供了比 synchronized 更丰富的功能,如可中断锁、公平锁、定时锁等

  • Read/Write Lock

    ReadWriteLock 接口定义了读写锁,它包含两个锁,一个用于读操作,一个用于写操作。读写锁允许多个线程同时读取共享资源,但只允许一个线程进行写操作。ReentrantReadWriteLockReadWriteLock 的一个实现类

    import java.util.concurrent.locks.ReadWriteLock;
    import java.util.concurrent.locks.ReentrantReadWriteLock;
    
    public class Demo {
        private final ReadWriteLock readWriteLock = new ReentrantReadWriteLock();
        public void read() {
            readWriteLock.readLock().lock();
            try {
                // 读取共享资源的操作
            } finally {
                readWriteLock.readLock().unlock();
            }
        }
        public void write() {
            readWriteLock.writeLock().lock();
            try {
                // 修改共享资源的操作
            } finally {
                readWriteLock.writeLock().unlock();
            }
        }
    }
    

    读写锁适用于读操作远远多于写操作的场景,可以提高并发性

  • StampedLock

    它是一种读写锁的变种,提供了乐观读锁,可以在读多写少的场景中提供更好的性能

    import java.util.concurrent.locks.StampedLock;
    
    public class Demo {
        private final StampedLock stampedLock = new StampedLock();
        public void read() {
            long stamp = stampedLock.tryOptimisticRead();
            // 乐观读操作
            if (!stampedLock.validate(stamp)) {
                // 有写操作发生,转为悲观读
                stamp = stampedLock.readLock();
                try {
                    // 悲观读操作
                } finally {
                    stampedLock.unlockRead(stamp);
                }
            }
        }
        public void write() {
            long stamp = stampedLock.writeLock();
            try {
                // 写操作
            } finally {
                stampedLock.unlockWrite(stamp);
            }
        }
    }
    

    StampedLock 提供了更细粒度的控制,并允许在不同的代码路径中执行不同的操作

总结

在多线程编程中,确保线程安全是至关重要的!通过合理使用 synchronized 关键字、volatile 关键字以及 ReentrantLock 等锁机制,可以有效地保护共享资源,避免数据不一致和竞态条件等问题。合理的同步机制不仅能够提高程序的性能,还能够确保程序的正确性。在实际开发中,根据具体场景选择合适的同步和锁机制是编写高效、安全多线程代码的关键。

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

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

相关文章

html和css写淘宝的快速导航条

目录 1、css代码 2、html代码 1、css代码 <style>* {margin: 0;padding: 0;list-style: none;text-decoration: none;}.nav {width: 900px;height: 40px;background-color: rgb(247, 249, 250);margin: 50px auto;padding-left: 30px;}.nav>li {float: left;width: 1…

软件测试中年危机?30岁大关?“我“该如何破局...

目录&#xff1a;导读 前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结&#xff08;尾部小惊喜&#xff09; 前言 在软件测试行业摸…

Oracle 表数据锁了,处理方式,Oracle锁表处理

Oracle 表数据锁了&#xff0c;处理方式&#xff0c;Oracle锁表处理 参考连接&#xff1a; Oracle锁表处理_oracle锁表怎么解决_辣椒炒鸡的博客-CSDN博客 实践&#xff1a; 1、查看被锁的表 select b.owner,b.object_name,a.session_id,a.locked_mode from v$locked_object …

QTimer的使用

参考&#xff1a; Qt 定时器 (QTimer)的几种使用方法-CSDN博客 QTimer 如何判断QT定时器在工作&#xff1f;-CSDN博客 (1)timeout信号 每隔5秒&#xff0c;timer发射timeout()信号&#xff0c;w执行onTimeout()槽函数。 Widget w;QTimer timer;timer.start(5*1000);QObjec…

点评项目——商户查询缓存

2023.12.7 redis实现商户查询缓存 在企业开发中&#xff0c;用户的访问量动辄成百上千万&#xff0c;如果没有缓存机制&#xff0c;数据库将承受很大的压力。本章我们使用redis来实现商户查询缓存。 原来的操作是根据商铺id直接从数据库查询商铺信息&#xff0c;为了防止频繁地…

WIFI7:O7851PM吞吐测试

近年来&#xff0c;随着科技的飞速发展&#xff0c;WIFI技术也在不断演进&#xff0c;从WIFI6逐渐过渡到WIFI6E&#xff0c;再到如今备受关注的WIFI7。这次我们将聚焦于我们欧飞信最新研发的WIFI7模组&#xff1a;O7851PM&#xff0c;通过对WIFI7的核心指标——吞吐量测试&…

IoT DC3 是一个基于 Spring Cloud 全开源物联网平台 linux docker部署傻瓜化步骤

如有不了解可先参考我的另一篇文章本地部署:IoT DC3 是一个基于 Spring Cloud 的开源的、分布式的物联网(IoT)平台本地部署步骤 如有不了解可先参考我的另一篇文章本地部署: 1 环境准备: JDK 8 以上 docker 安装好 下载docker-compose-dev.yml 文件 执行基础环境docker安装 …

如何使用技术 SEO 优化 Pinterest 富图钉

Pinterest 可以影响搜索引擎排名&#xff0c;尤其是谷歌。不过&#xff0c;它的作用方式与其他搜索引擎优化因素不同。这就是 Google 将图钉放在 nofollow 列表中。但是&#xff0c;它们仍然可以作为搜索引擎优化的一个重要因素。 高质量的图钉具有高分辨率的图片、吸引人的内…

【高数:1 映射与函数】

【高数&#xff1a;1 映射与函数】 例2.1 绝对值函数例2.2 符号函数例2.3 反函数表示例2.4 双曲正弦sinh&#xff0c;双曲余弦cosh&#xff0c;双曲正切tanh 参考书籍&#xff1a;毕文斌, 毛悦悦. Python漫游数学王国[M]. 北京&#xff1a;清华大学出版社&#xff0c;2022. 例2…

【Hung-Yi Lee】强化学习笔记

文章目录 What is RLPolicy GradientPolicy Gradient实际是怎么做的On-policy v.s. Off-policyExploration配音大师 Actor-Critic训练value function的方式网络设计DQN Reward ShapingNo Reward&#xff1a;Learning from Demonstration What is RL 定义一个策略网络&#xff0…

JM中ref_pic_list_modification bug记录

问题描述 今天在用JM对YUV420p编码时,发现编出的码流用ffplay播放花屏,报如下错误: JM的版本时19.1,没有使能B帧,PicOrderCntType设置为2,其它都是encoder.cfg中的默认配置。我用一些码流分析工具播放H264码流正常,用一些播放器播放也都存在花屏,不过大多数播放器都是…

Java程序员,你掌握了多线程吗?

文章目录 01 多线程对于Java的意义02 为什么Java工程师必须掌握多线程03 Java多线程使用方式04 如何学好Java多线程写作末尾 摘要&#xff1a;互联网的每一个角落&#xff0c;无论是大型电商平台的秒杀活动&#xff0c;社交平台的实时消息推送&#xff0c;还是在线视频平台的流…

word中如何插入公式,如何高效使用mathtype,遇到他人论文的复杂公式如何直接粘贴复制,为你一一答疑解惑!!

文章目录 一、论文中插入公式---最原始&#xff0c;最好用&#xff0c;最稳定的方法1.1 主页--插入---对象1.2 找到公式编辑器&#xff0c;对应你的版本1.3 直接输入公式&#xff0c;关闭界面 二、如何做好一个懒人——如何直接粘贴别人PDF等格式论文中的公式&#xff1f;2.1 使…

活动回顾|德州仪器嵌入式技术创新发展研讨会(上海站)成功举办,信驰达科技携手TI推动技术创新

2023年11月28日&#xff0c;德州仪器(TI)嵌入式技术创新发展研讨会在上海顺利举办。作为TI中国第三方IDH&#xff0c;深圳市信驰达科技有限公司受邀参加&#xff0c;并设置展位&#xff0c;展出CC2340系列低功耗蓝牙模块及TPMS、蓝牙数字钥匙解决方案&#xff0c;与众多业内伙伴…

种下一棵栀子花

女孩说自己是男孩当年伸手就可触及的栀子花.男孩沉默了,明白了当初失去了什么. 时光倒流,13年前的一个夏天夜晚,男孩与女孩同在沪城,女孩分享了自己的照片给男孩.男孩站在窗前,那晚的夜空很美丽,仿佛托着那个白衣裳女孩的纯真的梦,来到了自己的身边. 女孩说彼此间只是少了一个深…

017 OpenCV 向量机SVM

目录 一、环境 二、SVM原理 三、完整代码 一、环境 本文使用环境为&#xff1a; Windows10Python 3.9.17opencv-python 4.8.0.74 二、SVM原理 OpenCV中的向量机&#xff08;SVM&#xff09;是一种监督学习算法&#xff0c;用于分类和回归分析。它通过找到一个最优的超平…

优化AI机器人外呼之话术制作

智能语音机器人&#xff0c;OKCC-AI厂家制作话术工作实践总结。 一、 语音机器人话术制作&#xff0c;一般可以分为三个模块&#xff1a; 第一&#xff1a;是主干流程&#xff0c;也叫主流程 这个部分有点类似我们人工呼叫中心&#xff0c;常说的话术脚本。 第二&#xff1a…

如何处理3dmax渲染完成后阴影部分?

使用3dmax软件&#xff0c;对效果图进行渲染过程中&#xff0c;有不少小伙伴&#xff0c;在渲染完成后出现问题。 较为常见的3dmax渲染问题有3dmax渲染有阴影&#xff1f; 对于一些新手伙伴遇到这类问题&#xff0c;不知如何解决&#xff0c;就会苦恼3dmax渲染有阴影怎么办&am…

剑指Offer 从尾到头打印链表

目录 刷题链接&#xff1a;题目描述思路一&#xff1a;复杂度分析python3C 刷题链接&#xff1a; https://www.nowcoder.com/practice/d0267f7f55b3412ba93bd35cfa8e8035 题目描述 输入一个链表的头节点&#xff0c;按链表从尾到头的顺序返回每个节点的值&#xff08;用数组返…

2024年AMC8考试报名的唯一途径和备考建议,别再浪费时间去探索

最近有些家长和孩子咨询六分成长2024年AMC8美国数学竞赛的报名和考试等事宜。 和他们交流&#xff0c;我发现很多家长和孩子走了许多弯路&#xff0c;被网上的信息带偏了&#xff0c;且浪费了很多时间。为此&#xff0c;六分成长将2024年AMC8报名和考试等的相关信息最新版&…