11. 深入理解并发编程-AQS与JMM

news/2024/11/20 21:44:52/

AQS (AbstractQueuedSynchronizer)
他的实现类诸如:
CountDownLatchThreadLocalPoolReentrantLock
在这些类中,AQS都是以内部类的形式存在的

AQS使用了模板方法设计模式
例子:
做蛋糕分为3个步骤,定一个抽象类,重写3个方法,做模型、烘焙和涂抹原料,然后在另外1个方法做蛋糕中,将这3个方法步骤依次放入;
然后具体的实现类继承自上面抽象类,比如做奶油蛋糕有奶油蛋糕的制作流程,做芝士蛋糕有芝士蛋糕的制作流程等等.

如何自己实现一个锁?
重写一个类MyLock实现Lock,然后重写Lock类中的lockunLock方法;然后重写1内部类MyAQS继承自AQS,然后重写AQS中的tryAcquire方法和tryRelease方法,这两个方法其实就是要更改AQSstate的状态:
AQSstate的状态改为1,即为获得锁(CAS实现,compareAndSetState(1, acquires)),同时调用setExclusiveOwnerThread(Thread.currentThread),设置当前线程获得锁;
AQSstate的状态改为改为0,即为释放锁,同时调用setExclusiveOwnerThread(null),当前线程释放锁;
最后在自定义的MyLock中的lockunLock方法中分别调用MyAQS.tryAcquire方法和MyAQS.tryRelease方法,即完成了一个自定义的Lock类.

AQS基本思想CLH队列锁
双链表实现,链表中每一个结点不停的去自旋(正常情况有次数限制,一般是2次,然后进入阻塞状态),查看前一个结点是否已经使用完锁并释放,前一个结点释放锁的同时当前结点的myPrednull,同时将当前结点的locked状态改为false;

了解ReentrantLock的实现

  1. 锁的可重入
    • 在递归或者是同步方法中包含了同步方法,就可能会出现锁的可重入问题,可以改造我们锁的state状态,如果发现是同一个线程在获取锁,state++;释放锁的时候state--
  2. 公平锁和非公平锁
    • 公平锁与非公平锁的区别就是这1行代码hasQueuedPredecessors(),公平锁会检测是否有等待队列,有等待队列则将新插入的线程添加到链表尾部;非公平锁则会与当前链表竞争锁

JMM (Java Memory Model 内存模型)

什么是工作内存和主内存?
主内存有点类似于JVM中的,而工作内存则类似于JVM中的.
正常情况定一个变量count,这个变量count是放在主内存的,而多个线程在对这个变量进行运算比如加1,这个操作过程是放在工作内存中完成的,每个工作内存都拥有一个这个主内存变量count的副本countCopy,并且每个子线程之间这个变量的副本是相互之间不可见的,当每个线程在运算完成后,会将这个变量副本countCopy赋值给主内存中的变量count(这里就是volatile可见性的作用,让别的子线程从主内存中拿到的这个变量值是最新的,而不是最初没有子线程操作过的那个旧的变量值).

JMM导致的并发安全问题?
count = count + 1;真的就只有一条语句吗?

  1. 可见性
    • 不同的线程之间做同一个变量的更改,可能存在可见性问题,不同线程更改变量副本后,主内存中变量的值可能不正确,可以采用加volitile(保证可见性)和(同时保证可见性和原子性)解决这个问题
  2. 原子性
    • 不同线程之间在更改同一个变量时,比如变量累加i++这个过程转换成字节码时,存在多行代码,在多行代码在执行的过程中,可能被别的线程打断,从而导致了别的线程处理完成后更改了主内存中变量的值,而被打断的线程继续执行字节码后面的逻辑,而没有从主内存中重新去取变量的值,导致了被打断的线程执行完成后,去更改主内存中变量的值,此时,这个是一个错误的值.

http://www.ppmy.cn/news/66749.html

相关文章

Android 13 有线网变更(用到的可以收藏)

Android 13 有线网变更 文章目录 Android 13 有线网变更一、从接触的Android13 (Tv版)源码和测试结果看,有线网有如下变更:二、有线网具体变更介绍1、限制了设置有线网参数设置接口方法2、新增有线网开启关闭接口方法3、新增了 up…

怎样恢复删除的视频?这5个方法才是正确答案!

案例:怎样恢复删除的视频? 【我是个视频爱好者,平常会在电脑中存很多视频,但也经常会将很多视频误删,怎样恢复删除的视频呢?希望大家给我一些建议!】 在摄影摄像技术较发达的今天,…

Python中__repr__和__str__方法的区别|留学生CS辅导帖

Python中__repr__和__str__方法的区别 你好,我是悦创。 问题 __repr__和__str__有什么区别? 解答 __repr__和__str__方法在Python中都用于为对象提供字符串表示形式,但它们之间存在一些区别,主要体现在用途和实现上&#xff…

队列、栈专题

队列、栈专题 LeetCode 20. 有效的括号解题思路代码实现 LeetCode 921. 使括号有效的最少添加解题思路代码实现 LeetCode 1541. 平衡括号字符串的最少插入次数解题思路代码实现 单调栈LeetCode 496. 下一个更大元素 I解题思路代码实现 LeetCode 739. 每日温度解题思路代码实现 …

一个网站引发的程序猿的牢骚,哈哈哈

2013年大学毕业后,参加工作做的第一个前端项目,北京服装学院,今天调研一个关于iframe的需求,突然想试试,以前那些做IE6兼容的项目是否还在使用,就默默的点开了。十年了,他们没有换网站&#xff…

【Pandas与SQL系列】Pandas实现分布函数percent_rank、cume_dist

目录 1,分布函数,1.1,percent_rank()1.2,cume_dist()1.3 SQL例子 2,Pandas 实现3,补充Pandas实现排序 1,分布函数, 应用场景:快速查看某个记录所归属的组内的比例 分布函数分类及基础语法&…

JAVA-代码块和内部类

文章目录 目录 文章目录 前言 1.代码块 1.1什么是代码块? 1.2代码块的分类及作用: 1.静态代码块 2.成员代码块(又叫做构造代码块) 3.局部代码块 2.内部类 2.1 什么是内部类? 2.2 内部类的分类 1.成员内部类 2.静态内部类 3.匿名内部类 4.局部内部类 总结 前言 作者简介:我是最…

PSP - AlphaFold2 适配不同来源搜索的 MSA 接口

欢迎关注我的CSDN:https://spike.blog.csdn.net/ 本文地址:https://blog.csdn.net/caroline_wendy/article/details/130594303 MSA (Multiple Sequence Alignment) 在 AlphaFold2 中的工作方式如下: 使用搜索工具 (hhblits/hhsearch/jackhmmer),从大型数据库中,搜索与目标…