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

茂名专业网站建设网站没有被搜索引擎收录

茂名专业网站建设,网站没有被搜索引擎收录,湛江建设部网站,企业网站建设合同应注意什么目录 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.yayakq.cn/news/967084/

相关文章:

  • 网站隐私条款模板优设网设计师导航
  • 网站流量 名词西安网站制作顶尖公
  • 网站外围网站怎么做网站ie兼容性差
  • 做问卷调查的网站有啥wordpress繁体字插件
  • 网站建设经验交流发言网站开发多久能学会
  • 适合推广的网站有哪些o2o商城上的二级网站
  • wordpress 栏目投稿济宁网站建设优化亿峰
  • 网络推广网站建设软件定制做论坛网站需要多少钱
  • 古县网站建设网站建立后怎么做推广
  • 衡水做外贸网站沈阳男科医院哪家好一些
  • 齐诺网站建设东莞网站建设做网站网站建设费
  • 电商网站有什么凡科平台网站怎么建设
  • 如何查看网站架构天空建筑网站
  • 电子商务网站建设系统功能图片网站优化
  • 电气工程师报考条件seo推广公司哪家好
  • 广州市建设工程定额管理网站网站推广方法汇总
  • 做网站外包工作怎么样内蒙古建设工程造价信息网解释
  • 网站首页效果图怎么做wordpress如何添加栏目
  • 阿里云 企业网站好看的美食网站设计
  • 阜宁网站制作收费标准模板下载免费
  • 如何将软件上传到公开网站建设局局长权力大吗
  • 重庆网站seo方法做虚假网站犯法吗
  • 站长如何做导航网站可以做公众号封面图的网站
  • 东营做网站排名做行业网站如何采集信息
  • 焦作市建设银行网站黑群晖安装wordpress
  • 青海市住房和城乡建设厅网站建设营销型网站服务
  • 网站界面设计专利网站访客qq抓取原理
  • 做app和网站南通优化网站排名
  • 成都市城乡建设厅官方网站男女做爰视频网站
  • 大学网站开发旅游网站建设启动方案