当前位置: 首页 > news >正文

安阳360网站推广工具爱站工具包

安阳360网站推广工具,爱站工具包,手机网站建设哪家公司好,在线做logo印章网站目录 CAS什么是CASCAS的应用ABA问题异常举例 Synchronized 原理基本特征加锁过程偏向锁轻量级锁重量级锁 其他优化操作锁消除锁粗化 CAS 什么是CAS CAS: 全称Compare and swap,字面意思:”比较并交换“,CAS涉及如下操作: 假设内存中的原数据…

目录

  • CAS
    • 什么是CAS
    • CAS的应用
    • ABA问题
      • 异常举例
  • Synchronized 原理
    • 基本特征
    • 加锁过程
      • 偏向锁
      • 轻量级锁
      • 重量级锁
    • 其他优化操作
      • 锁消除
      • 锁粗化

CAS

什么是CAS

CAS: 全称Compare and swap,字面意思:”比较并交换“,CAS涉及如下操作:
假设内存中的原数据为A,旧的预期值为B ,需要修改的值为C。

  1. 首先把A与B进行比较,看A与B是否相同。
  2. 如果A与B相同,则把数据C的值赋予A。
  3. 返回操作成功。

我们来写一个CAS的伪代码以帮忙我们更好理解CAS。

 boolean Cas(int a,int b,int c){//进行比较看a是否发生变化if(a==b){a=c;return true;}return false;}

CAS是乐观锁的一种实现方式,当多个线程对一个数据进行操作时,只有一个线程操作成功,其他线程并不会阻塞,会返回操作失败的信号。
真实的 CAS 是一个原子的硬件指令完成的,只有硬件予以支持,软件方面才能实现。

CAS的应用

标准库中提供了 java.util.concurrent.atomic 包,里面的类都是基于这种方式来实现的。
典型的就是 AtomicInteger 类, 其中的 getAndIncrement 相当于 i++ 操作。

public static void main(String[] args) {ReentrantReadWriteLock lock = new ReentrantReadWriteLock();AtomicInteger  seq = new AtomicInteger(0);//进行++操作seq.getAndIncrement();seq.getAndIncrement();seq.getAndIncrement();System.out.println(seq);}

在这里插入图片描述
我们点开自增方法,我们看到它的操作也是通过上述伪代码的那种方式实现的。
在这里插入图片描述
也可以使用CAS实现自旋锁

ABA问题

假设存在两个线程 t1 和 t2。 有一个共享变量 num, 初始值为 A。
接下来,线程 t1 想使用 CAS 把 num 值改成 Z,那么就需要

  • 先读取 num 的值, 记录到 oldNum 变量中。
  • 使用 CAS 判定当前 num 的值是否为 A, 如果为 A,就修改成 Z。
    但是,在 t1 执行这两个操作之间,t2 线程可能把 num 的值从 A 改成了 B, 又从 B 改成了 A。

异常举例

以银行取钱为例:

  1. 存款 100,线程1 获取到当前存款值为 100,期望更新为 50; 线程2 获取到当前存款值为 100, 期望更新为 50。
  2. 线程1 执行扣款成功, 存款被改成 50。线程2 阻塞等待中。
  3. 在线程2 执行之前, 你的朋友正好给你转账 50, 账户余额变成,100。
  4. 轮到线程2 执行了,发现当前存款为 100,和之前读到的 100 相同,再次执行扣款操作。

这样我们的钱就不翼而飞了,所以这种情况是万万不可的。

所以我们引入版本号来解决这个问题。CAS在读取旧值时也要读取版本号,在修改时,如果读到的版本号与当前版本号相同就进行修改,如果当前版本号高于读到的版本号,就修改失败。

Synchronized 原理

基本特征

  1. 开始时是乐观锁,如果锁冲突严重就升级为悲观锁。
  2. Synchronized是可重入锁。
  3. 是不公平锁。
  4. 是不可读写锁
  5. 开始是轻量级锁实现,如果锁被持有的时间较长, 就转换成重量级锁。

加锁过程

加锁流程图:
在这里插入图片描述

偏向锁

偏向锁就是在当前锁对象中标记改锁属于那个线程,没有进行实际加锁,能不加锁就不加锁,减少不必要的开销,只有当其他线程来竞争锁时,才会进行锁升级,由偏向锁变为轻量级锁。

轻量级锁

锁升级为轻量级锁之后,通过CAS实现。

  • 通过CAS检查并更新一块内存。
  • 如果更新成功,则认为加锁成功。
  • 如果更新失败,则认为加锁失败,锁被占用

重量级锁

如果竞争进一步激烈, 自旋不能快速获取到锁状态,就会膨胀为重量级锁
此处的重量级锁就是指用到内核提供的 mutex。

  • 执行加锁操作, 先进入内核态。
  • 在内核态判定当前锁是否已经被占用
  • 如果该锁没有占用, 则加锁成功,并切换回用户态。
  • 如果该锁被占用,则加锁失败。 此时线程进入锁的等待队列,挂起。 等待被操作系统唤醒。
  • 经历了一系列的沧海桑田, 这个锁被其他线程释放了, 操作系统也想起了这个挂起的线程, 于是唤醒这个线程, 尝试重新获取锁。

当多个线程竞争同一把锁,自旋等待的时间过长,无法获取到锁时,JVM会将这把锁升级为重量级锁。这时,线程并不再进行自旋等待,而是进入内核态,通过操作系统提供的mutex实现来管理锁的状态和等待队列。
在内核态中,操作系统判定当前锁是否已经被占用。如果锁没有被占用,则线程成功获取到锁,并切换回用户态继续执行。如果锁已经被占用,则线程加锁失败。此时,线程会进入锁的等待队列,并被操作系统挂起,等待被唤醒。
随着时间的推移和线程的竞争,当其他线程释放了这把锁并且操作系统意识到有线程在等待这个锁时,操作系统会唤醒等待的线程,使其重新启动并尝试重新获取锁。这个过程可能会经历一段时间,之后线程再次尝试获取锁以继续执行。

其他优化操作

锁消除

编译器+JVM 判断锁是否可消除,如果可以,就直接进行消除了。
也就是说我们许多加锁操作在单线程中运行时,那些加锁操作的锁就没必要。

 @Overridepublic synchronized StringBuffer append(String str) {toStringCache = null;super.append(str);return this;}

例如 StringBuffe中的append操作就会涉及加锁操作,我们在单线程运行中就可以进行锁消除。

锁粗化

一段逻辑中如果出现多次加锁解锁,编译器 + JVM 会自动进行锁的粗化。

用我们上课讲的例子就是:

领导给下面人布置任务呢,一共三个任务,现在有这两种做法:

  1. 给员工打一个电话一次性什么三个任务。
  2. 给员工打三个电话,一次说一个任务。

让我们大家选择,大家肯定选择做法一啊,当然人家jvm也会进行这样的锁粗化。

可以用一个代码理解一下:

        //频繁加锁for (int i = 0; i < 100; i++) {synchronized (o1){}}//粗化synchronized (o1){for (int i = 0; i < 100; i++) {}}

把锁粗化,避免频繁申请释放锁。

http://www.dinnco.com/news/64666.html

相关文章:

  • 怎样注册.com网站大数据培训机构排名前十
  • 网站做跳转的意义统计站老站长推荐草莓
  • 仙桃建设网站年轻人不要做网络销售
  • wordpress wp百度网站优化培训
  • 太原建站公司有哪些天津快速关键词排名
  • 企业网站布局960舆情网站直接打开的软件
  • 成都科技网站建设费网销平台排名
  • 做中文网站公司微信群免费推广平台
  • 长沙网站搭建首选智投未来windows优化大师值得买吗
  • 南宁营销型网站公司网站建设推广
  • dede 子网站企业营销模式
  • 紫砂壶网站开发与设计报告论文电脑培训网上课程
  • 乐山做网站北京百度推广优化
  • 外链网站分类微信小程序开发费用
  • 沧州网站建设开发服务北京计算机培训机构哪个最好
  • 简洁大气的企业网站搜索引擎是网站吗
  • 微信网站打不开近三天的国内外大事
  • 网购网站后台管理系统图片南昌搜索引擎优化
  • 网站案例库nba最新交易一览表
  • 江门网站推广技巧服务google官方下载
  • 网站怎么做流量统计百度广告联盟app
  • 佛山建设网站公司互联网营销培训课程
  • 企业建设银行网站登录不了百度收录关键词
  • 淘宝网站都是怎么做的吗成都seo的方法
  • 南京网站开发南京乐识权威新媒体运营主要做什么
  • 个体工商户可以申请网站建设吗站长工具排行榜
  • 营销网站建设大概费用网站的推广
  • 无锡网站制作百度知道网址
  • cdr做网站分辨率搜索引擎优化代理
  • 公众号设置下载wordpress网站优化 seo和sem