0

0

java并发编程(8)原子变量和非阻塞的同步机制

巴扎黑

巴扎黑

发布时间:2017-06-26 09:18:48

|

1662人浏览过

|

来源于php中文网

原创

原子变量和非阻塞的同步机制

 一、锁的劣势

  1.在多线程下:锁的挂起和恢复等过程存在着很大的开销(及时现代的jvm会判断何时使用挂起,何时自旋等待)

  2.volatile:轻量级别的同步机制,但是不能用于构建原子复合操作

  因此:需要有一种方式,在管理线程之间的竞争时有一种粒度更细的方式,类似与volatile的机制,同时还要支持原子更新操作

二、CAS

  独占锁是一种悲观的技术--它假设最坏的情况,所以每个线程是独占的

  而CAS比较并交换:compareAndSwap/Set(A,B):我们认为内存处值是A,如果是A,将其修改为B,否则不进行操作;返回内存处的原始值或是否修改成功

立即学习Java免费学习笔记(深入)”;

  如:模拟CAS操作 

//模拟的CASpublic class SimulatedCAS {private int value;public synchronized int get() {return value;
    }//CAS操作public synchronized int compareAndSwap(int expectedValue, int newValue) {int oldValue = value;if (oldValue == expectedValue) {
            value = newValue;
        }return oldValue;
    }public synchronized boolean compareAndSet(int expectedValue, int newValue) {return (expectedValue == compareAndSwap(expectedValue, newValue));
    }
}//典型使用场景public class CasCounter {private SimulatedCAS value;public int getValue() {return value.get();
    }public int increment() {int v;do {
            v = value.get();
        } while {
            (v != value.compareAndSwap(v, v + 1));
        }return v + 1;
    }
}

 

  JAVA提供了CAS的操作

    原子状态类:AtomicXXX的CAS方法

    JAVA7/8:对Map的操作:putIfAbsent、computerIfAbsent、computerIfPresent.........

三、原子变量类

   AtomicRefence原子更新对象,可以是自定义的对象;如:

public class CasNumberRange {private static class IntPair {// INVARIANT: lower <= upperfinal int lower;        //将值定义为不可变域final int upper;        //将值定义为不可变域public IntPair(int lower, int upper) {this.lower = lower;this.upper = upper;
        }
    }private final AtomicReference values = new AtomicReference(new IntPair(0, 0));    //封装对象public int getLower() {return values.get().lower;
    }public int getUpper() {return values.get().upper;
    }public void setLower(int i) {while (true) {
            IntPair oldv = values.get();if (i > oldv.upper) {throw new IllegalArgumentException("Can't set lower to " + i + " > upper");
            }
            IntPair newv = new IntPair(i, oldv.upper);  //属性为不可变域,则每次更新新建对象if (values.compareAndSet(oldv, newv)) {     //原子更新,如果在过程中有线程修改了,则其他线程不会更新成功,因为oldv与内存处值就不同了return;
            }
        }
    }//同上public void setUpper(int i) {while (true) {
            IntPair oldv = values.get();if (i < oldv.lower)throw new IllegalArgumentException("Can't set upper to " + i + " < lower");
            IntPair newv = new IntPair(oldv.lower, i);if (values.compareAndSet(oldv, newv))return;
        }
    }
}

 

  性能问题:使用原子变量在中低并发(竞争)下,比使用锁速度要快,一般情况下是比锁速度快的

 

四、非阻塞算法

  许多常见的数据结构中都可以使用非阻塞算法

  非阻塞算法:在多线程中,工作是否成功有不确定性,需要循环执行,并通过CAS进行原子操作

MCP官网
MCP官网

Model Context Protocol(模型上下文协议)

下载

  1、上面的CasNumberRange

  2、栈的非阻塞算法:只保存头部指针,只有一个状态

//栈实现的非阻塞算法:单向链表public class ConcurrentStack  {
    AtomicReference> top = new AtomicReference>();public void push(E item) {
        Node newHead = new Node(item);
        Node oldHead;do {
            oldHead = top.get();
            newHead.next = oldHead;
        } while (!top.compareAndSet(oldHead, newHead));//CAS操作:原子更新操作,循环判断,非阻塞    }public E pop() {
        Node oldHead;
        Node newHead;do {
            oldHead = top.get();if (oldHead == null) {return null;
            }
            newHead = oldHead.next;
        } while (!top.compareAndSet(oldHead, newHead));//CAS操作:原子更新操作,循环判断,非阻塞return oldHead.item;
    }private static class Node  {public final E item;public Node next;public Node(E item) {this.item = item;
        }
    }
}

 

  3、链表的非阻塞算法:头部和尾部的快速访问,保存两个状态,更加复杂

public class LinkedQueue  {private static class Node  {final E item;final AtomicReference> next;public Node(E item, LinkedQueue.Node next) {this.item = item;this.next = new AtomicReference>(next);
        }
    }private final LinkedQueue.Node dummy = new LinkedQueue.Node(null, null);private final AtomicReference> head = new AtomicReference>(dummy);private final AtomicReference> tail = new AtomicReference>(dummy);  //保存尾节点public boolean put(E item) {
        LinkedQueue.Node newNode = new LinkedQueue.Node(item, null);while (true) {
            LinkedQueue.Node curTail = tail.get();
            LinkedQueue.Node tailNext = curTail.next.get();if (curTail == tail.get()) {if (tailNext != null) {// 处于中间状态,更新尾节点为当前尾节点的next                    tail.compareAndSet(curTail, tailNext);
                } else {// 将当前尾节点的next 设置为新节点:链表if (curTail.next.compareAndSet(null, newNode)) {/** * 此处即为中间状态,虽然在这里进行了两次原子操作,整体不是原子的,但是通过算法保证了安全:
                         * 原因是处于中间状态时,如果有其他线程进来操作,则上面那个if将执行;
                         * 上面if的操作是来帮助当前线程完成更新尾节点操作,而当前线程的更新就会失败返回,最终则是更新成功                         */// 链接成功,尾节点已经改变,则将当前尾节点,设置为新节点                        tail.compareAndSet(curTail, newNode);return true;
                    }
                }
            }
        }
    }
}

 

  3.原子域更新器

    上面的逻辑,实现了链表的非阻塞算法,使用Node来保存头结点和尾节点

    在实际的ConcurrentLinkedQueue中使用的是基于反射的AtomicReferenceFiledUpdater来包装Node

五、ABA问题

  CAS操作中容易出现的问题:

    判断值是否为A,是的话就继续更新操作换为B;

    但是如果一个线程将值A改为C,然后又改回A,此时,原线程将判断A=A成功执行更新操作;

    如果把A改为C,然后又改回A的操作,也需要视为变化,则需要对算法进行优化

  解决:添加版本号,每次更新操作都要更新版本号,即使值是一样的

      

 

相关专题

更多
php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

65

2025.12.31

php网站源码教程大全
php网站源码教程大全

本专题整合了php网站源码相关教程,阅读专题下面的文章了解更多详细内容。

43

2025.12.31

视频文件格式
视频文件格式

本专题整合了视频文件格式相关内容,阅读专题下面的文章了解更多详细内容。

35

2025.12.31

不受国内限制的浏览器大全
不受国内限制的浏览器大全

想找真正自由、无限制的上网体验?本合集精选2025年最开放、隐私强、访问无阻的浏览器App,涵盖Tor、Brave、Via、X浏览器、Mullvad等高自由度工具。支持自定义搜索引擎、广告拦截、隐身模式及全球网站无障碍访问,部分更具备防追踪、去谷歌化、双内核切换等高级功能。无论日常浏览、隐私保护还是突破地域限制,总有一款适合你!

41

2025.12.31

出现404解决方法大全
出现404解决方法大全

本专题整合了404错误解决方法大全,阅读专题下面的文章了解更多详细内容。

204

2025.12.31

html5怎么播放视频
html5怎么播放视频

想让网页流畅播放视频?本合集详解HTML5视频播放核心方法!涵盖<video>标签基础用法、多格式兼容(MP4/WebM/OGV)、自定义播放控件、响应式适配及常见浏览器兼容问题解决方案。无需插件,纯前端实现高清视频嵌入,助你快速打造现代化网页视频体验。

9

2025.12.31

关闭win10系统自动更新教程大全
关闭win10系统自动更新教程大全

本专题整合了关闭win10系统自动更新教程大全,阅读专题下面的文章了解更多详细内容。

8

2025.12.31

阻止电脑自动安装软件教程
阻止电脑自动安装软件教程

本专题整合了阻止电脑自动安装软件教程,阅读专题下面的文章了解更多详细教程。

3

2025.12.31

html5怎么使用
html5怎么使用

想快速上手HTML5开发?本合集为你整理最实用的HTML5使用指南!涵盖HTML5基础语法、主流框架(如Bootstrap、Vue、React)集成方法,以及无需安装、直接在线编辑运行的平台推荐(如CodePen、JSFiddle)。无论你是新手还是进阶开发者,都能轻松掌握HTML5网页制作、响应式布局与交互功能开发,零配置开启高效前端编程之旅!

2

2025.12.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 2.2万人学习

C# 教程
C# 教程

共94课时 | 5.7万人学习

Java 教程
Java 教程

共578课时 | 40.4万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号