JUC——原子类操作(三)
原子类操作
既然强调了并发访问,那么就必须考虑操作系统位数:32位操作系统还是64位操作系统,对于long型数据类型而言,是64位的。但是如果现在项目运行在32位系统上,则long型数据会占用32位空间进行数据的保存。

如果现在每一个程序类里面都去使用long类型,那么进行处理的时候都需要手动进行volatile配置,那样就太麻烦了。
为了解决这样的问题,在juc里面提供了一个atomic子包,这个子包里面保存的都是原子性的操作数据,也就是说里面包含的数据类型都使用volatile进行声明。
原子操作分类
原子操作:是指操作过程中不会被中断,保证数据操作是以原子方式进行的。
- 基本数据类型:AtomicInteger, AtomicLong, AtomicBoolean
- 数组类型:AtomicIntegerArray, AtomicLongArray, AtomicRefernceArray
- 引用类型: AtomicReference, AtomicStampedRerence, AtomicMarkableReference
- 对象的属性修改类型: AtomicIntegerFieldUpdater, AtomicLongFieldUpdater, AtomicReferenceFieldUpdater.
范例:观察“AtomicLong”类型
实际上在使用“AtomicLong”的时候里面包含的就是“private volatile long value”.
package so.strong.concurrents;
import java.util.concurrent.atomic.AtomicLong;
public class StrongDemo {
public static void main(String[] args) throws Exception {
AtomicLong num = new AtomicLong(10);
System.out.println("数据自增:"+num.incrementAndGet());
System.out.println("数据自减:"+num.decrementAndGet());
}
}
如果要进行数据类型的基础数学运算也是需要通过方法进行的。
范例:实现基础数学运算
package so.strong.concurrents;
import java.util.concurrent.atomic.AtomicLong;
public class StrongDemo {
public static void main(String[] args) throws Exception {
AtomicLong num = new AtomicLong(100);
System.out.println("加法操作:"+num.addAndGet(10));
System.out.println("减法操作:"+num.addAndGet(-9));
}
}
毕竟这种操作不是原始的基本数据类型,它的操作时刻需要保证数据在多线程访问下的并发安全性。
对于原子性的处理,以上并不是它的重点,只是它的操作形式,这里面有一个最为重要的方法,CAS方法:
public final boolean compareAndSet(long expect, long update) {
return unsafe.compareAndSwapLong(this, valueOffset, expect, update);
}
范例:观察CAS方法的使用
package so.strong.concurrents;
import java.util.concurrent.atomic.AtomicLong;
public class StrongDemo {
public static void main(String[] args) throws Exception {
AtomicLong num = new AtomicLong(100); //设置原子性操作
//如果现在要进行修改的内容是100,即:原始的原子类型里面为100,则使用300替换num的内容
System.out.println(num.compareAndSet(101,300)); //比较的值等于100,返回true
System.out.println(num);
}
}
使用CAS方法进行内容修改的时候一定要设置一个原始的比较内容,如果内容相同才可以修改,如果现在操作的是数组也有与之对应的程序类AtomicLongArray。
AtomicLongArray有两个构造方法:
- 动态开辟:设置数组的长度
public AtomicLongArray(int length) {
array = new long[length];
} - 静态开辟:设置具体的数组内容
public AtomicLongArray(long[] array) {
this.array = array.clone();
}
范例:进行数组操作
package so.strong.concurrents;
import java.util.concurrent.atomic.AtomicLongArray;
public class StrongDemo {
public static void main(String[] args) throws Exception {
AtomicLongArray array = new AtomicLongArray(new long[]{1,2,3});
array.set(0,99); //原子性的数组必须使用set修改内容
System.out.println(array);
}
}
除了对long类型进行原子性的处理支持之外也可以对引用类型(对象)进行原子性操作。
范例:使用原子性进行对象的描述
package so.strong.concurrents;
import java.util.concurrent.atomic.AtomicReference;
public class StrongDemo {
public static void main(String[] args) throws Exception {
AtomicReference<Member> ref = new AtomicReference<>();
Member memA = new Member("张三", 22);
Member memB = new Member("李四", 32);
ref.set(memA);
ref.compareAndSet(memA, memB); //对象引用变更只得依靠地址比较“==”
System.out.println(ref);
}
} class Member {
private String name;
private int age;
public Member(String name, int age) {
this.name = name;
this.age = age;
} @Override
public String toString() {
return "name= " + this.name + ",age= " + this.age;
}
}
以上的几种类型严格来讲都算是常用的几种处理形式,但是在Java开发里面有可能会遇见一种很奇怪的问题,即:可能本身类中定义的类型不是AtomicLong,那么可以利用AtomicLongFieldUpdater类来完成。
public abstract class AtomicLongFieldUpdater<T>
// 抽象类AtomicLongFieldUpdater更新器
AtomicLongFieldUpdater更新器获得对象的方法:
@CallerSensitive
public static <U> AtomicLongFieldUpdater<U> newUpdater(Class<U> tclass, String fieldName) {
Class<?> caller = Reflection.getCallerClass();
if (AtomicLong.VM_SUPPORTS_LONG_CAS)
return new CASUpdater<U>(tclass, fieldName, caller);
else
return new LockedUpdater<U>(tclass, fieldName, caller);
}
范例:使用AtomicLongFieldUpdater更新器
package so.strong.concurrents;
import java.util.concurrent.atomic.AtomicLongFieldUpdater;
public class StrongDemo {
public static void main(String[] args) throws Exception {
Book book = new Book(100001,"Java从入门到放弃");
book.setBid(200002); //修改bid
System.out.println(book);
}
} class Book {
private volatile long bid; //必须追加volatile关键字
private String title;
public Book(long bid, String title) {
this.bid = bid;
this.title = title;
} @SuppressWarnings({"rawtypes","unchecked"})
public void setBid(long bid) {
AtomicLongFieldUpdater updater = AtomicLongFieldUpdater.newUpdater(super.getClass(), "bid");
updater.compareAndSet(this, this.bid, bid);//使用CAS方法进行内容的修改
} @Override
public String toString() {
return "图书编号: " + this.bid + ", 名称: " + this.title;
}
}
并发访问中为了保证多位的数据类型的完整性一定要使用volatile关键字,同时在整个juc的开包里面会发现有大量的原子操作类出现。
JUC——原子类操作(三)的更多相关文章
- java 多线程系列---JUC原子类(三)之AtomicLongArray原子类
AtomicLongArray介绍和函数列表 在"Java多线程系列--“JUC原子类”02之 AtomicLong原子类"中介绍过,AtomicLong是作用是对长整形进行原子操 ...
- Java多线程系列--“JUC原子类”02之 AtomicLong原子类
概要 AtomicInteger, AtomicLong和AtomicBoolean这3个基本类型的原子类的原理和用法相似.本章以AtomicLong对基本类型的原子类进行介绍.内容包括:Atomic ...
- Java多线程系列--“JUC原子类”04之 AtomicReference原子类
概要 本章对AtomicReference引用类型的原子类进行介绍.内容包括:AtomicReference介绍和函数列表AtomicReference源码分析(基于JDK1.7.0_40)Atomi ...
- Java多线程系列--“JUC原子类”05之 AtomicLongFieldUpdater原子类
概要 AtomicIntegerFieldUpdater, AtomicLongFieldUpdater和AtomicReferenceFieldUpdater这3个修改类的成员的原子类型的原理和用法 ...
- 【Java_多线程并发编程】JUC原子类——4种原子类
根据修改的数据类型,可以将JUC包中的原子操作类可以分为4种,分别是: 1. 基本类型: AtomicInteger, AtomicLong, AtomicBoolean ;2. 数组类型: Atom ...
- java多线程系类:JUC原子类:03之AtomicLongArray原子类
概要 AtomicIntegerArray, AtomicLongArray, AtomicReferenceArray这3个数组类型的原子类的原理和用法相似.本章以AtomicLongArray对数 ...
- Java多线程系列--“JUC原子类”03之 AtomicLongArray原子类
概要 AtomicIntegerArray, AtomicLongArray, AtomicReferenceArray这3个数组类型的原子类的原理和用法相似.本章以AtomicLongArray对数 ...
- JDK原子类操作
JDK原子类操作及原理 在JDK5之后,JDK提供了对变量的原子类操作, java.util.concurrent.atomic里都是原子类 原子类的分类 原子更新基本类型 原子更新数组 原子更新抽象 ...
- 【Java_多线程并发编程】JUC原子类——原子类中的volatile变量和CAS函数
JUC中的原子类是依靠volatile变量和Unsafe类中的CAS函数实现的. 1. volatile变量的特性 内存可见性(当一个线程修改volatile变量的值后,另一个线程就可以实时看到此变量 ...
随机推荐
- PAT——1040. 有几个PAT
字符串APPAPT中包含了两个单词“PAT”,其中第一个PAT是第2位(P),第4位(A),第6位(T):第二个PAT是第3位(P),第4位(A),第6位(T). 现给定字符串,问一共可以形成多少个P ...
- [AHOI2001]多项式乘法
\([Link](https://www.luogu.org/problemnew/show/P2553)\) \(\color{red}{\mathcal{Description}}\) 给出两个多 ...
- 用PSCP在Windows和Linux之间相互传输文件
在Linux服务器之间相互传文件我们常用 scp命令,但是在Linux和Windows之间相互传输就不那么直接了. 使用 Putty的 PSCP 则会简单的多 1. 下载 http://www.chi ...
- POJ 1384 Intervals (区间差分约束,根据不等式建图,然后跑spfa)
传送门: http://acm.hdu.edu.cn/showproblem.php?pid=1384 Intervals Time Limit: 10000/5000 MS (Java/Others ...
- vlc源码分析(一) RTSP会话流程
可以先了解一下RTSP/RTP/RTCP的概念与区别:RTP与RTCP协议介绍(转载). 在调试vlc-android时,熟悉了RTSP的会话流程.C表示RTSP客户端,S表示RTSP服务端: 第一步 ...
- 【Dubbo源码阅读系列】之 Dubbo SPI 机制
最近抽空开始了 Dubbo 源码的阅读之旅,希望可以通过写文章的方式记录和分享自己对 Dubbo 的理解.如果在本文出现一些纰漏或者错误之处,也希望大家不吝指出. Dubbo SPI 介绍 Java ...
- Xcode官方xip直接离线下载地址(更新到Xcode 9.4.1)
Xcode 9.4.1 https://download.developer.apple.com/Developer_Tools/Xcode_9.4.1/Xcode_9.4.1.xip Xcode 9 ...
- MySQL学习【第十一篇存储引擎之事务解释】
一.innodb的核心特点------事务 1.什么是事务 在一组数据操作执行步骤,这些步骤被视为一个单元,主要针对dml语句(update.delete.insert) 2.事务ACID特性 Ato ...
- Ubuntu下apt方式安装与更新Git
本人使用的系统 Ubuntu 18.04.1 ,使用apt安装Git: sudo apt insatll git 安装后发现不是最新的版本,更新方法: sudo add-apt-repository ...
- 偏前端-HTML5 sessionStorage-会话存储
sessionStorage 是HTML5新增的一个会话存储对象,用于临时保存同一窗口(或标签页)的数据,在关闭窗口或标签页之后将会删除这些数据.本篇主要介绍 sessionStorage(会话存储) ...