上一节我们说到了基本原子类的简单介绍,这一节我们先来看一下基本类型: AtomicInteger, AtomicLong, AtomicBoolean。AtomicInteger和AtomicLong的使用方法差不多,AtomicBoolean因为比较简单所以方法比前两个都少,那我们这节主要挑AtomicLong来说,会使用一个,其余的大同小异。

1.原子操作与一般操作异同

我们在说原子操作之前为了有个对比为什么需要这些原子类而不是普通的基本数据类型就能满足我们的使用要求,那就不得不提原子操作不同的地方。

当你在操作一个普通变量时,你在Java实现的每个操作,在程序编译时会被转换成几个机器能读懂的指令。例如,当你分配一个值给变量,在Java你只使用了一个指令,但是当你编译这个程序时,这个指令就被转换成多个JVM 语言指令。这样子的话当你在操作多个线程且共享一个变量时,就会导致数据不一致的错误。

为了避免这样的问题,Java引入了原子变量。当一个线程正在操作一个原子变量时,即使其他线程也想要操作这个变量,类的实现中含有一个检查那步骤操作是否完成的机制。 基本上,操作获取变量的值,改变本地变量值,然后尝试以新值代替旧值。如果旧值还是一样,那么就改变它。如果不一样,方法再次开始操作。这个操作称为 Compare and Set(简称CAS,比较并交换的意思)。

原子变量不使用任何锁或者其他同步机制来保护它们的值的访问。他们的全部操作都是基于CAS操作。它保证几个线程可以同时操作一个原子对象也不会出现数据不一致的错误,并且它的性能比使用受同步机制保护的正常变量要好。

2.AtomicLong简介

由字面意义我们可以知道AtomicLong可以用原子方式更新的 long 值,下面我们看一下他的构造方法和一般方法:

构造方法:

AtomicLong() //创建具有初始值 0 的新 AtomicLong。

AtomicLong(long initialValue) //创建具有给定初始值的新 AtomicLong。

方法:

long addAndGet(long delta) //以原子方式将给定值添加到当前值。

boolean compareAndSet(long expect, long update) //如果当前值 == 预期值,则以原子方式将该值 设置为给定的更新值。

long decrementAndGet() //以原子方式将当前值减 1。

double doubleValue() //以 double 形式返回指定的数值。

float floatValue() //以 float 形式返回指定的数值。

long get() //获取当前值。

long getAndAdd(long delta) //以原子方式将给定值添加到当前值。

long getAndDecrement() //以原子方式将当前值减 1。

long getAndIncrement() //以原子方式将当前值加 1。

long getAndSet(long newValue)// 以原子方式设置为给定值,并返回旧值。

long incrementAndGet() //以原子方式将当前值加 1。

int intValue() // 以 int 形式返回指定的数值。

void lazySet(long newValue) //最后设置为给定值。

long longValue() // 以 long 形式返回指定的数值。

void set(long newValue) //设置为给定值。

String toString() // 返回当前值的字符串表示形式。

boolean weakCompareAndSet(long expect, long update) //如果当前值 == 预期值,则以原子方式将该值设置为给定的更新值。

3.使用AtomicLong

3.1   创建AtomicLong

创建AtomicLong的过程如下:

AtomicLong atomicLong=new AtomicLong();

此示例创建一个初始值为0的AtomicLong 。

如果你想创建一个带有初始值的AtomicLong ,你可以这样做:

AtomicLong atomicLong=new AtomicLong(123);

此示例将值123作为参数传递给AtomicLong装订器,该装置将AtomicLong实例的初始值设置为123 。

3.2   获取AtomicLong值

您可以通过get()方法get() AtomicLong实例的值。 这里是一个AtomicLong.get()示例:

AtomicLong atomicLong=new AtomicLong(123);

long theValue = atomicLong.get();

设置AtomicLong值

您可以通过set()方法set() AtomicLong实例的值。 这里是一个AtomicLong.set()示例:

AtomicLong atomicLong = new AtomicLong(123);

atomicLong.set(234);

此示例创建一个初始值为123的AtomicLong示例,然后在下一行中将其值设置为234 。

3.3   比较并设置AtomicLong值

AtomicLong类也有一个原子compareAndSet()方法。 此方法将AtomicLong实例的当前值与AtomicLong进行比较,如果这两个值相等, AtomicLong实例设置新值。 这里是一个AtomicLong.compareAndSet()示例:

AtomicLong atomicLong = new AtomicLong(123);

long expectedValue = 123;

long newValue = 234;

atomicLong.compareAndSet(expectedValue,newValue);

此示例首先创建一个初始值为123的AtomicLong实例。 然后,它将AtomicLong的值与期望值123进行比较,如果它们相等,则AtomicLong的新值变为234 ;

3.4   添加到AtomicLong值

AtomicLong类包含几个方法,您可以使用这些方法向AtomicLong添加值并返回其值。这里我们要重点关注一下,因为这几个方法会如果我们使用不当会造成歧义。 这些方法是:

addAndGet()

getAndAdd()

getAndIncrement()

incrementAndGet()

第一种方法addAndGet()向AtomicLong添加一个数字,并在添加后返回其值。

第二种方法getAndAdd()还向AtomicLong添加一个数字,但返回AtomicLong在添加值之前的值。

您应该使用这两种方法中的哪一种取决于您的用例。 这里有两个例子:

AtomicLong atomicLong = new AtomicLong();

System.out.println(atomicLong.getAndAdd(10));

System.out.println(atomicLong.addAndGet(10));

此示例将打印出值0和20 。 首先,示例在添加10之前获取AtomicLong的值。 它的值在加法之前为0.然后示例将10添加到AtomicLong ,并获取添加后的值。 该值现在为20。

您也可以通过这两种方法向AtomicLong添加负数。 结果实际上是一个减法。

方法getAndIncrement()和incrementAndGet()工作原理像getAndAdd()和addAndGet()但只是添加1到AtomicLong的值。

3.5   从AtomicLong值中减去

AtomicLong类还包含一些用于从AtomicLong值中以AtomicLong值的方法。 这些方法是:

decrementAndGet()

getAndDecrement()

decrementAndGet()从AtomicLong值中减去1,并在AtomicLong后返回其值。 getAndDecrement()也从AtomicLong值中减去1,但返回AtomicLong在AtomicLong之前的值。

由上我们大致知道了AtomicLong的用法,AtomicBoolean,AtomicInteger也与它的用法差不多,我们看一下API他们各自的方法就知道该如何使用。

java 原子数据类型_java并发编程(十一)----(JUC原子类)基本类型介绍相关推荐

  1. 【Java_多线程并发编程】JUC原子类——4种原子类

    根据修改的数据类型,可以将JUC包中的原子操作类可以分为4种,分别是: 1. 基本类型: AtomicInteger, AtomicLong, AtomicBoolean ; 2. 数组类型: Ato ...

  2. java volatile 原子性_Java并发编程之验证volatile不能保证原子性

    Java并发编程之验证volatile不能保证原子性 通过系列文章的学习,凯哥已经介绍了volatile的三大特性.1:保证可见性 2:不保证原子性 3:保证顺序.那么怎么来验证可见性呢?本文凯哥(凯 ...

  3. java内存 海子_Java并发编程:从根源上解析volatile关键字的实现

    Java并发编程:volatile关键字解析 1.解析概览 内存模型的相关概念 并发编程中的三个概念 Java内存模型 深入剖析volatile关键字 使用volatile关键字的场景 2.内存模型的 ...

  4. java投票锁_Java并发编程锁之独占公平锁与非公平锁比较

    Java并发编程锁之独占公平锁与非公平锁比较 公平锁和非公平锁理解: 在上一篇文章中,我们知道了非公平锁.其实Java中还存在着公平锁呢.公平二字怎么理解呢?和我们现实理解是一样的.大家去排队本着先来 ...

  5. java lock 对象_Java并发编程锁系列之ReentrantLock对象总结

    Java并发编程锁系列之ReentrantLock对象总结 在Java并发编程中,根据不同维度来区分锁的话,锁可以分为十五种.ReentranckLock就是其中的多个分类. 本文主要内容:重入锁理解 ...

  6. java计算时间差_JAVA并发编程三大Bug源头(可见性、原子性、有序性),彻底弄懂...

    原创声明:本文转载自公众号[胖滚猪学编程]​ 某日,胖滚猪写的代码导致了一个生产bug,奋战到凌晨三点依旧没有解决问题.胖滚熊一看,只用了一个volatile就解决了.并告知胖滚猪,这是并发编程导致的 ...

  7. java 延迟初始化_Java并发编程——延迟初始化占位类模式

    --仅作笔记使用,内容多摘自<java并发编程实战> 在并发编程中,如果状态变量仅在单个线程中初始化和使用,自然是线程安全的,但一旦涉及到线程间的数据交互,如何声明一个用于多线程的单例状态 ...

  8. java 线程工厂_Java并发编程:Java的四种线程池的使用,以及自定义线程工厂

    引言 通过前面的文章,我们学习了Executor框架中的核心类ThreadPoolExecutor ,对于线程池的核心调度机制有了一定的了解,并且成功使用ThreadPoolExecutor 创建了线 ...

  9. java线程池_Java 并发编程 线程池源码实战

    作者 | 马启航 杏仁后端工程师.「我头发还多,你们呢?」 一.概述 笔者在网上看了好多的关于线程池原理.源码分析相关的文章,但是说实话,没有一篇让我觉得读完之后豁然开朗,完完全全的明白线程池,要么写 ...

最新文章

  1. ExecutorService 的理解与使用
  2. C# 中的委托和事件
  3. TMB计算的小工具-calculate_TMB.exe
  4. java 执行cd_Java调用Linux命令(cd的处理)
  5. python 相关性分析_数据分析---用Python进行相关性分析(兼谈假设检验)
  6. Java script第二课
  7. c语言变量为什么要定义,C语言为什么要规定对所用到的变量要“先定义,后使用”...
  8. 记一次 .NET 某云采购平台API 挂死分析
  9. 自定义PopView
  10. java 电子编号生成器_业务编号生成器
  11. 《数据分析实战:基于EXCEL和SPSS系列工具的实践》一3.4 数据量太大了怎么办
  12. __attribute__((visibility()))
  13. SSM药店管理系统的设计与实现答辩PPT模板
  14. HDU4607 Park Visit(解法二)【废除!!!】
  15. 55. 安全 HTTP(3)
  16. spring bean作用域之间有什么区别
  17. 1072. 开学寄语
  18. 如何往ncbi上上传数据
  19. matlab 中trace,trace命令和tracert命令的用途 matlab中的trace 怎么用的?
  20. Gif录制神器GifCam

热门文章

  1. mysql存储过程知识点_知识点:Mysql 基本用法之存储过程
  2. mnesia mysql性能_Mnesia
  3. python如何画曲线图_如何使用python画曲线图
  4. mac nginx加载php 配置,Mac下Nginx安装环境配置详解
  5. python 动态相册_Python编程:制作电子相册
  6. android软键盘把布局顶上去,Android 软键盘弹出时把原来布局顶上去的解决方法
  7. java font.getfont_java – 无法解析方法’getFont(?)’
  8. modbus_百度经验
  9. iOS博客 视频课程网站
  10. 响应式网页设计简单入门