Java并发控制 学习笔记1

news2025/1/9 16:31:46

一、并发控制的方法

1、悲观锁:常用的互斥锁都属于悲观锁,一个线程访问共享资源时其他线程不能访问。

2、乐观锁:允许同时访问共享数据,只有在提交时利用如版本号检查是否有冲突,应用github。

3、什么时候用乐观锁、什么时候用悲观锁?

写操作比较多的时候悲观锁,读操作多用乐观锁。再就是一个事务执行时间很长时用乐观锁,冲突很多时考虑悲观锁。

二、什么是自旋锁?

正常的锁当一个线程拿不到共享资源时,会把该线程阻塞,等共享资源释放后,再唤醒该阻塞线程并进行调度;

自旋锁当一个线程拿不到共享资源时,它就一直while循环来询问该资源释放了没有,会一直占用cpu资源。

适用情况:如果大多事务的执行时间很短,自旋锁空转一会就能拿到资源,就用自旋锁,这时如果采用阻塞/唤醒思路的话,调度的时间消耗会比较大。

三、代码实现:

例1(leetcode1114按序打印):

方案1(synchronized关键字):

        既可以实现自旋锁,也可以实现阻塞/唤醒思路,但要注意自旋锁时很容易死锁,所以synchronized包裹哪一部分代码考虑清楚,同时若出现“不可见”问题,考虑适用volatile关键字。

代码(自旋锁):

class Foo {

    private volatile Integer flag = 0;

    public Foo() {
        
    }

    public void first(Runnable printFirst) throws InterruptedException {
        
// 1、正常synchronized解决了“不可见问题”,
// 但因为这里flag!=0在synchronized代码块外边,
// 所以这里的flag用的是“线程存储空间中的值”,
// 不是主存中的最新值,所以使用volatile来定义flag变量;

// 2、另外synchronized(this)为什么不是synchronized(flag),
// 因为Integer flag是引用类型,改变flag的值,flag指向的存储空间就变了,
// 作为共享资源的变量存储空间是一直不能变的。
        while (flag !=0) {} 
        synchronized(this) {
            printFirst.run();
            flag = 1;
        }
        
    }

    public void second(Runnable printSecond) throws InterruptedException {
        
            
        while (flag != 1) {}
        synchronized(this) {
            System.out.println(".." + flag);
            printSecond.run();
            flag = 2;
        }
        
    }

    public void third(Runnable printThird) throws InterruptedException {
        
        while (flag != 2);
        synchronized(this) {
            System.out.println(flag);
            flag = 0;
            printThird.run();
        }

    }
}


///
synchronized少包裹一些代码也行,如下:

class Foo {

    private volatile Integer flag = 0;

    public Foo() {
        
    }

    public void first(Runnable printFirst) throws InterruptedException {
        
        
        while (flag !=0) {}
        
        printFirst.run();
        synchronized(this) {
            flag = 1;
        }
        
    }

    public void second(Runnable printSecond) throws InterruptedException {
        
            
        while (flag != 1) {}

        printSecond.run();
        synchronized(this) {
            flag = 2;
        }
        
    }

    public void third(Runnable printThird) throws InterruptedException {
        
        while (flag != 2);
        printThird.run();
        synchronized(this) {
            flag = 0;
        }

    }
}

代码(阻塞/唤醒思路):

class Foo {

    private Integer flag = 0;

    public Foo() {
        
    }

    public void first(Runnable printFirst) throws InterruptedException {
        // 可以用this,也可以另外定义一个一定不会被修改的Object,
        // 但不能用flag,原因如上代码。
        synchronized(this) {
            while (flag !=0) {this.wait();}
            printFirst.run();
            flag = 1;
            this.notifyAll();
        }
        
    }

    public void second(Runnable printSecond) throws InterruptedException {
        synchronized(this) {
            
            while (flag != 1) {this.wait();}
            printSecond.run();
            flag = 2;
            this.notifyAll();
        }
        
    }

    public void third(Runnable printThird) throws InterruptedException {
        synchronized(this) {
            while (flag != 2) this.wait();
            flag = 0;
            printThird.run();
            this.notifyAll();
        }

    }
}

方案2(Lock类):和synchronized关键字差不多,只不过是一个封装好的类,还是得借助volatile关键字实现解决可见性,下面代码是自旋锁思路,也可以用条件锁Condition类实现阻塞思路,如文章最后的链接中的方法(synchronized配合wait等方法  == Lock配合Condition,只不过前者更底层);

class Foo {

    private volatile Integer flag;
    private Lock lock = new ReentrantLock();

    public Foo() {
        flag = 0;
    }

    public void first(Runnable printFirst) throws InterruptedException {
        
        while (flag != 0) {}
        lock.lock();
        printFirst.run();
        flag = 1;
        lock.unlock();
    }

    public void second(Runnable printSecond) throws InterruptedException {
        while (flag != 1);   
        lock.lock();     
        printSecond.run();
        flag = 2;
        lock.unlock();
    }

    public void third(Runnable printThird) throws InterruptedException {
        while (flag != 2);
        lock.lock();
        printThird.run();
        flag = 0;
        lock.unlock();
    }
}

方案3(原子类AtomicInteger类):

也是既可以实现自旋锁,也可以实现阻塞唤醒,AtomicInteger不用=、!=、+等运算符,要调用get()方法、set()方法、incrementAndGet()等方法。

自旋锁代码:

class Foo {
    
    private AtomicInteger flag = new AtomicInteger(0);

    public Foo() {
        System.out.println(flag);
    }

    public void first(Runnable printFirst) throws InterruptedException {
        
        while (flag.get() != 0);
        printFirst.run();
        flag.set(1);
    }

    public void second(Runnable printSecond) throws InterruptedException {
        
        while (flag.get() != 1);
        printSecond.run();
        flag.set(2);
    }

    public void third(Runnable printThird) throws InterruptedException {
        
        while (flag.get() != 2);
        printThird.run();
        flag.set(0);
    }
}

阻塞/唤醒思路代码:

使用xx.wait、xx.notifyALL函数时,xx一定得对于所有线程可见。xx可以是另外定义的一个量,并配合synchronized关键字实现对所有线程可见,或者不用synchronized用volatile修饰一下也可以。

对于AtomicInteger类 阻塞不好实现,wait、notifyAll还是配合synchronized关键字用。

方案4(信号量、计数器、阻塞队列等,这些方面实现阻塞/唤醒思路,同时因为这些工具类都是线程安全的,都能满足“可见性”,所以只要这些工具类能有类似AtomicInteger类get/set方法的方法,那也能实现自旋锁,不过我还没试过...),下面链接都属阻塞/唤醒思路的:

链接:力扣icon-default.png?t=N2N8https://leetcode.cn/problems/print-in-order/solutions/488685/si-chong-bu-tong-de-shi-xian-fang-shi-zong-jie-by-/

end...

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

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

相关文章

开发必备:EsayCode使用以及Oracle自定义模板

前言 写前先问一句,不会还有人在手动写这些基础的sql语句吧?! 最近在做Oracle的项目,手写mapper和entity文件真是写到手软,以前MySQL都是找的线上自动生成的,现在也不行了。 找了很长时间,也…

【Python】tkinter的简单使用(Tk对象、三大布局、变量、事件)

本文目录1.tkinter2.Tk对象3.三大布局3.1 pack布局3.2 grid布局3.3 place布局4.变量5.事件1.tkinter tkinter是Tcl/Tk GUI工具包(即使用Tcl语言开发Tk图形库)的标准Python接口,支持在Windows、macOS、Linux多平台运行。 tkinter是Python自带…

DPDK入门(环境搭建以及小demo)

文章目录零、从0开始配置dpdk环境的虚拟机一、dpdk的编译usertool/dpdk-setup.sh二、dpdk需要什么配置来支持1.多队列网卡2.巨页三、解析接收网络数据的过程经历了什么1.物理网卡2.NIC3.内核协议栈4.标准接口层Posix API5. 应用层上述过程发生的拷贝四、DPDK介绍基于上述接收网…

Github采用Http Push失败

Github采用Http Push失败 Github的密码凭证从2021年起开始就不能用了,现在采用http去push代码时候提示输入的密码要换成令牌(token)才可以。 如何在Github上生成自己的令牌呢? (1)简单来说就是将原来输入…

Linux搭建GitLab私有仓库,并内网穿透实现公网访问

文章目录前言1. 下载Gitlab2. 安装Gitlab3. 启动Gitlab4. 安装cpolar5. 创建隧道配置访问地址6. 固定GitLab访问地址6.1 保留二级子域名6.2 配置二级子域名7. 测试访问二级子域名前言 GitLab 是一个用于仓库管理系统的开源项目,使用Git作为代码管理工具&#xff0c…

游戏运营专员的职责有哪些?提高游戏收入的关键是什么?

游戏运营是将一款游戏平台推入市场,通过对平台的运作,使用户从接触、认识、再到了解实际线上的一种操作、最终成为这款游戏平台的忠实玩家的这一过程。同时通过一系列的营销手段达到提高线上人数,刺激消费增长利润的目的。 游戏运营专员的职…

4.3 协方差及相关系数、矩

学习目标: 我正在学习协方差、相关系数和矩,我会采取以下措施: 理解基本概念:首先,我会努力理解协方差、相关系数和矩的基本概念。我会查阅参考资料,例如课本或在线教程,以便深入了解这些概念…

Unity设计模式—子类沙盒

Unity设计模式—子类沙盒 一个基类定义了一个抽象的沙河方法和一些预定义的操作集合。通过将它们设置为受保护的状态已确定它们仅供子类使用。每个派生的沙盒子类根据父类提供的操作来实现沙盒操作。 子类沙盒的名字比较生僻,其实内容非常常见,平常用的很…

警惕,3月20日WOS目录更新,50本SCI/SSCI被剔除,这个出版社多达18本

2023年3月SCI、SSCI期刊目录更新 2023年3月20日,Web of Science核心期刊目录再次更新!此次2023年3月SCIE & SSCI期刊目录更新,与上次更新(2023年2月)相比,共有50本期刊被剔除出SCIE & SSCI期刊目录…

香橙派4LTS和树莓派4B构建K8S集群实践之一:K8S安装

目录 1. 说明 1.1 软硬件环境 1.2 设计目标 2 实现 2.1 准备工作 - 香橙派 (k8s-master-1) - 树莓派 (k8s-node-1) - 两派都要干的事 2.2 containerd 安装与设置 2.3 安装 3 遇到的问题 3.1 k8s-master-1 3.2 k8s-node-1 4 相关命令 5 Tips 6 参考 1. 说明 …

【论文笔记】Deformable Convolutional Networks

1.介绍 1.1. 提出DCN的原因 卷积神经网络(CNNs)由于其固定几何结构局限了模型几何变换。 1.2.本文核心贡献:提出了两种新模块 deformable convolution 和 deformable RoI pooling 第一种是可变形卷积。它将2D偏移添加到标准卷积中的规则网…

【点云概述】什么是点云、来源、种类、特点、处理?

文章目录 一、什么是点云二、点云的来源三、点云的种类四、点云的特点五、点云的处理一、什么是点云 点云(point cloud)是空间中点的数据集,可以表示三维形状或对象,通常由三维扫描仪获取。点云中每个点的位置都由一组笛卡尔坐标 ( X , Y , Z ) (X, Y, Z) (X,Y,Z)描述,有…

【C++】4. 重载函数

文章目录 前言一、函数重载1.1 重载的条件1.2 函数名修饰 前言 自然语言中,一个词可以有多重含义,人们可以通过上下文来判断该词真实的含义,即该词被重载了。比如:以前有一个笑话,国有两个体育项目大家根本不用看&…

双目三维测距(python)

文章目录 1. 双目检测1.1 调用相机1.2 分割画面 2. 双目标定2.1 相机标定2.2 获取参数 3. 双目测距3.1 立体校正3.1.1 校正目的3.1.2 校正方法3.1.2 相关代码 3.2 立体匹配和视差计算3.3 深度计算3.4 注意事项 4. 完整代码 代码打包下载: 链接1:https://…

Spring MVC 的调用(12)

目录 SpringMVC流程 源码分析 第一步:用户发起请求到前端控制器(DispatcherServlet) 第二步:前端控制器请求处理器映射器(HandlerMappering)去查找处理器(Handle):通过xml配置或者…

LeetCode:28. 找出字符串中第一个匹配项的下标 ——【1、理解 KMP 算法】

🍎道阻且长,行则将至。🍓 🌻算法,不如说它是一种思考方式🍀算法专栏: 👉🏻123 目录一、🌱[28. 找出字符串中第一个匹配项的下标](https://leetcode.cn/proble…

synchronized 的 monitor 机制

synchronized 的 monitor 机制 前言 本文基于 jdk 8 编写。author JellyfishMIX - github / blog.jellyfishmix.comLICENSE GPL-2.0 monitor monitor 是 synchronized 中用以实现线程之间的互斥与协作的主要手段,它可以看成是对象或者 class 持有的锁。每一个对象…

十六、市场活动:查看市场活动明细(一)

功能需求 点击市场活动名称链接,跳转到明细页面,查看市场活动明细 -市场活动的基本信息 -市场活动下所有的备注信息 功能分析 流程图 代码实现 一、ActivityMapper 1.ActivityMapper接口 /*** 点击名称,查看市场详细*/Activity selectActivityForDetailById(Stri…

相量的加减乘除计算

相量的加减乘除计算 矢量是物理学中的术语,是指具有大小(magnitude)和方向的量。如速度、加速度、力等等就是这样的量。向量是数学中的术语,也称为欧几里得向量、几何向量、矢量。与向量对应的量叫做数量,在物理学中称…

二叉树基础概念

1.二叉树种类 1.1 满二叉树 满二叉树:如果一棵二叉树只有度为 0 0 0 的结点和度为 2 2 2 的结点,并且度为0的结点在同一层上,则这棵二叉树为满二叉树。 如图所示: 这棵二叉树为满二叉树,也可以说深度为 k k k&…