Java并发编程实战 第15章 原子变量和非阻塞同步机制
非阻塞的同步机制
简单的说,那就是又要实现同步,又不使用锁。
与基于锁的方案相比,非阻塞算法的实现要麻烦的多,但是它的可伸缩性和活跃性上拥有巨大的优势。
实现非阻塞算法的常见方法就是使用volatile语义和原子变量。
硬件对并发的支持
原子变量的产生主要是处理器的支持,最重要的是大多数处理器架构都支持的CAS(比较并交换)指令。
模拟实现AtomicInteger的++操作
首先我们模拟处理器的CAS语法,之所以说模拟,是因为CAS在处理器中是原子操作直接支持的。不需要加锁。
- public synchronized int compareAndSwap(int exceptValue, int newValue){
- int oldValue = count;
- if(oldValue == exceptValue){
- count = newValue;
- }
- return oldValue;
- }
注意:CAS总是返回oldValue。
使用上面的方法模拟AtomicInteger的++操作。
- class MyAtomicInteger{
- private int value;
- public int incrementAndGet()
- {
- int v ;
- do{
- v = value;
- }while(v != compareAndSwap(v,v+1));
- return v + 1;
- }
- }
注意:Java的AtomicInteger大概实现机制就是这样的,不会阻塞,使用处理器的CAS功能,但是要轮询尝试。
看起来轮询尝试性能会更差,其实不然,当竞争不是非常高的时候,基于CAS的算法更胜一筹。
原子类
AtomicBoolean AtomicInteger AtomicLong AtomicReference
原子数组类:AtomicIntegerArray AtomicLong AtomicReferenceArray。
使用volatile语法修饰的数组只能保证数组变量本身的volatile语义,不能保证元素的volatile语义。这个时候应该使用,原子数组类。
注意:AtomicBoolean AtomicInteger AtomicLong和非原子的对应数值类如Integer截然不用。实现机制完全不一样,也没有对应关系。最重要的一个差别:这个三个原子类是可变的。而且是使用的Object的hashCode和equals方法,没有自己扩展。
性能比较:锁与原子变量
在中低程度的竞争下,原子变量能提供很高的可伸缩性,原子变量性能超过锁;而在高强度的竞争下,锁能够更有效地避免竞争,锁的性能将超过原子变量的性能。但在更真实的实际情况(一般没有那么高强大的竞争)中,原子变量的性能将超过锁的性能。
注意:不论是锁还是原子变量,都远远比不上避免共享状态(如使用线程封闭技术,但是使用场景计较局限),彻底消除竞争的效率。
两个非阻塞的算法示例
- /**
- * 使用Treiber算法构造的非阻塞栈
- */
- public class ConcurrentStack<E> {
- private AtomicReference<Node<E>> top = new AtomicReference<ConcurrentStack.Node<E>>();
- public void push(E item){
- Node<E> newHead = new Node<E>(item);
- Node<E> oldHead;
- do{
- oldHead = top.get();
- newHead.next = oldHead;
- } while (!top.compareAndSet(oldHead, newHead));
- }
- public E pop(){
- Node<E> oldHead;
- Node<E> newHead;
- do {
- oldHead = top.get();
- if (oldHead == null)
- return null;
- newHead = oldHead.next;
- } while (!top.compareAndSet(oldHead, newHead));
- return oldHead.item;
- }
- private static class Node<E>{
- public final E item;
- public Node<E> next;
- public Node(E item){
- this.item = item;
- }
- }
- }
下面这个没有看懂,留在这里记录,以后再看:
- /**
- * 链表中非阻塞算法中的插入排序,来自Michael-Scott
- */
- public class LinkedQueue<E> {
- private static class Node<E>{
- final E item;
- final AtomicReference<Node<E>> next;
- public Node(E item, Node<E> next){
- this.item = item;
- this.next = new AtomicReference<>(next);
- }
- }
- private final Node<E> dummy = new Node<E>(null, null);
- private final AtomicReference<Node<E>> head =
- new AtomicReference<>(dummy);
- private final AtomicReference<Node<E>> tail =
- new AtomicReference<>(dummy);
- public boolean put(E item){
- Node<E> newNode = new Node<E>(item, null);
- while (true){
- Node<E> curTail = tail.get();
- Node<E> tailNext = curTail.next.get();
- if (curTail == tail.get()){ //尾部还未修改
- if (tailNext != null){
- // 队列处于中间状态(即新节点已经接上,尾节点还未更新),推进尾节点
- tail.compareAndSet(curTail, tailNext);
- } else{
- // 处于稳定状态, 尝试插入新节点
- if (curTail.next.compareAndSet(null, newNode)){
- // 插入成功后,推进尾节点
- tail.compareAndSet(curTail, tailNext);
- return true;
- }
- }
- }
- }
- }
- }
Java并发编程实战 第15章 原子变量和非阻塞同步机制的更多相关文章
- 《Java并发编程实战》第十五章 原子变量与非堵塞同步机制 读书笔记
一.锁的劣势 锁定后假设未释放.再次请求锁时会造成堵塞.多线程调度通常遇到堵塞会进行上下文切换,造成很多其它的开销. 在挂起与恢复线程等过程中存在着非常大的开销,而且通常存在着较长时间的中断. 锁可能 ...
- Java多线程并发编程之原子变量与非阻塞同步机制
1.非阻塞算法 非阻塞算法属于并发算法,它们可以安全地派生它们的线程,不通过锁定派生,而是通过低级的原子性的硬件原生形式 -- 例如比较和交换.非阻塞算法的设计与实现极为困难,但是它们能够提供更好的吞 ...
- Java并发编程实战---第六章:任务执行
废话开篇 今天开始学习Java并发编程实战,很多大牛都推荐,所以为了能在并发编程的道路上留下点书本上的知识,所以也就有了这篇博文.今天主要学习的是任务执行章节,主要讲了任务执行定义.Executor. ...
- Java并发编程实战 第16章 Java内存模型
什么是内存模型 JMM(Java内存模型)规定了JVM必须遵循一组最小保证,这组保证规定了对变量的写入操作在何时将对其他线程可见. JMM为程序中所有的操作定义了一个偏序关系,称为Happens-Be ...
- java并发编程实战:第十五章----原子变量与非阻塞机制
非阻塞算法:使用底层的原子机器指令(例如比较并交换指令)代替锁来确保数据在并发访问中的一致性 应用于在操作系统和JVM中实现线程 / 进程调度机制.垃圾回收机制以及锁和其他并发数据结构 可伸缩性和活跃 ...
- 【java并发编程实战】第一章笔记
1.线程安全的定义 当多个线程访问某个类时,不管允许环境采用何种调度方式或者这些线程如何交替执行,这个类都能表现出正确的行为 如果一个类既不包含任何域,也不包含任何对其他类中域的引用.则它一定是无状态 ...
- java并发编程实战:第二章----线程安全性
一个对象是否需要是线程安全的取决于它是否被多个线程访问. 当多个线程访问同一个可变状态量时如果没有使用正确的同步规则,就有可能出错.解决办法: 不在线程之间共享该变量 将状态变量修改为不可变的 在访问 ...
- Java并发编程实战 第8章 线程池的使用
合理的控制线程池的大小: 下面内容来自网络.不过跟作者说的一致.不想自己敲了.留个记录. 要想合理的配置线程池的大小,首先得分析任务的特性,可以从以下几个角度分析: 任务的性质:CPU密集型任务.IO ...
- 《Java并发编程实战》第二章 线程安全性 读书笔记
一.什么是线程安全性 编写线程安全的代码 核心在于要对状态訪问操作进行管理. 共享,可变的状态的訪问 - 前者表示多个线程訪问, 后者声明周期内发生改变. 线程安全性 核心概念是正确性.某个类的行为与 ...
随机推荐
- mysql中文乱码 常见编码问题解决方法分享
我是真的服了 mysql默认字符不是utf-8也不是GBK而是拉丁文字?? 在增删数据时 “中文字符” 老是乱码不停!害得我浪费不少时间在这上面 为各位之后不走坑 再此留下解决方法 若想进一步了解编码 ...
- 为解决Thymeleaf数字格式化问题而想到的几种方案
背景: spring后端输出double类型数据,前端使用thymeleaf框架,格式化double数据类型,由于需要显示原值(比如原来录入5,而不能显示5.00),因此需要存储数值(存储值为deci ...
- line-height 与 height 的区别
line-height是行高的意思,它决定了元素中文本内容的高度,height则是定义元素自身的高度. height:表示 行高 line-height:表示 每行文字所占的高度 举例: 第 ...
- linux是什么与如何学习(三)
1.1Linux是什么 Linux是在计算机上面运作的,所以说是一组软件. 1.2 Linux是什么?操作系统还是应用程序? 计算机主机是由一套硬件所组成的,为了有效的控制这些硬件资源,于是就有了操 ...
- elasticsearch-analysis-ik windows 环境 IK 中文分词器 的 下载 和 安装
1,下载插件压缩包(本地测试建议用迅雷下,生产用的绝对不要用迅雷下),链接地址:https://github.com/medcl/elasticsearch-analysis-ik/releases/ ...
- [百度知道]ssm和ssh各自的优势
https://zhidao.baidu.com/question/875108451824176892.html SSM和SSH不同主要在MVC实现方式,以及ORM持久化方面不同(Hiibernat ...
- ubuntu 安装 Anaconda2和3的tips
Anaconda 2 3 安装tips 安装anaconda2 我们要下载Anaconda2-4.3.0-Linux-x86_64.sh安装文件 下载好之后,在文件路径下执行以下命令: bash An ...
- gcc 数据对齐之:总结篇.
通过上面的分析,总结结构体对齐规则如下: 1.数据成员对齐规则:结构(struct)(或联合(union))的数据成员,第一个数据成员放在offset为0的地方,以后每个数据成员的对齐按照#pragm ...
- python中判断变量的类型
python的数据类型有:数字(int).浮点(float).字符串(str),列表(list).元组(tuple).字典(dict).集合(set) 一般通过以下方法进行判断: 1.isinstan ...
- python-day12(正式学习)
目录 可变长参数 可变长形参之* 可变长实参之* 可变长形参之** 可变长实参之** 可变长参数应用 命名关键字形参 函数对象 四大功能 引用 当作参数传给一个函数 可以当作函数的返回值 可以当作容器 ...