上一节我们说到了基本原子类的简单介绍,这一节我们先来看一下基本类型: 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并发编程(十一)----(JUC原子类)基本类型介绍的更多相关文章

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

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

  2. java并发编程(十二)----(JUC原子类)数组类型介绍

    上一节我们介绍过三个基本类型的原子类,这次我们来看一下数组类型: AtomicIntegerArray, AtomicLongArray, AtomicReferenceArray.其中前两个的使用方 ...

  3. 【Java_多线程并发编程】JUC原子类——原子类中的volatile变量和CAS函数

    JUC中的原子类是依靠volatile变量和Unsafe类中的CAS函数实现的. 1. volatile变量的特性 内存可见性(当一个线程修改volatile变量的值后,另一个线程就可以实时看到此变量 ...

  4. java并发编程-12个原子类

    背景 多线程更新变量的值,可能得不到预期的值,当然增加syncronized关键字可以解决线程并发的问题. 这里提供另外一种解决问题的方案,即位于 java.util.concurrent.atomi ...

  5. 【Java_多线程并发编程】JUC原子类——AtomicLong原子类

    1. AtomicLong是基本原子类中的一种 AtomicLong是对长整形进行原子操作. 1.1 AtomicLong类的函数列表 // 构造函数 AtomicLong() // 创建值为init ...

  6. 【Java并发工具类】原子类

    前言 为保证计数器中count=+1的原子性,我们在前面使用的都是synchronized互斥锁方案,加锁独占访问的方式未免太过霸道,于是我们来介绍另一种解决原子性问题的无锁方案:原子变量.在正式介绍 ...

  7. Java并发编程(3) JUC中的锁

    一 前言 前面已经说到JUC中的锁主要是基于AQS实现,而AQS(AQS的内部结构 .AQS的设计与实现)在前面已经简单介绍过了.今天记录下JUC包下的锁是怎么基于AQS上实现的 二 同步锁 同步锁不 ...

  8. Java并发编程原理与实战十三:JDK提供的原子类原理与使用

    原子更新基本类型 原子更新数组 原子更新抽象类型 原子更新字段 原子更新基本类型:   package com.roocon.thread.t8; import java.util.concurren ...

  9. java多线程系类:JUC原子类:03之AtomicLongArray原子类

    概要 AtomicIntegerArray, AtomicLongArray, AtomicReferenceArray这3个数组类型的原子类的原理和用法相似.本章以AtomicLongArray对数 ...

随机推荐

  1. 究竟什么是Windows句柄

    图解说明——究竟什么是Windows句柄     这里需要说明: 1.这里将句柄所能标识的所有东西(如窗口.文件.画笔等)统称为“对象”. 2.图中一个小横框表示一定大小的内存区域,并不代表一个字节, ...

  2. 牛逼了,教你用九种语言在JVM上输出HelloWorld

    我们在<深入分析Java的编译原理>中提到过,为了让Java语言具有良好的跨平台能力,Java独具匠心的提供了一种可以在所有平台上都能使用的一种中间代码——字节码(ByteCode). 有 ...

  3. [Poi2012]Festival 题解

    [Poi2012]Festival 时间限制: 1 Sec  内存限制: 64 MB 题目描述 有n个正整数X1,X2,...,Xn,再给出m1+m2个限制条件,限制分为两类: 1. 给出a,b (1 ...

  4. Shell文件

    #!/bin/bashecho "======================欢迎来到进爷故事会======================="echo "******* ...

  5. Centos6.5安装Redis3.2.8

    1 - Redis安装 redis安装 在网上一搜一大把,但是还是在这里想要能够统一吧,所以这个安装步骤是在Centos6.5 Minimal 上安装redis3.4.8,本次安装是在root 用户下 ...

  6. .NET多线程之Thread、Task、ThreadPool、Timer

    下表为多线程操作常见对象: 对象 方法/属性 描述 用途 用法 性能 Thread(线程) Start 启动线程,启动后线程处于System.Threading.ThreadState.Running ...

  7. Golang 高效实践之并发实践context篇

    前言 在上篇Golang高效实践之并发实践channel篇中我给大家介绍了Golang并发模型,详细的介绍了channel的用法,和用select管理channel.比如说我们可以用channel来控 ...

  8. 使用C#调试Windows服务模板项目

    Windows服务是非常强大的应用程序,可用于在backgorund中执行许多不同类型的任务.他们可以在不需要任何用户登录的情况下启动,并且可以使用除登录用户之外的其他用户帐户运行.但是,如果通过遵循 ...

  9. RabbitMQ从入门到精通(三)

    目录 1. 自定义消费者使用 自定义消费端演示 2.消费端的限流策略 2.1 限流的场景与机制 2.2 限流相关API 2.3 限流演示 3. 消费端ACK与重回队列机制 3.1 ACK与NACK 3 ...

  10. 个人永久性免费-Excel催化剂功能第48波-拆分工作薄内工作表,堪称Excel界的单反

    一个工作薄有多个相同类型的工作表,然后想通过批量操作,把每个工作表都另存为一个工作薄文件,这个批量拆分工作薄,绝大多数插件都有此功能,就如懂点VBA的高级用户也常常有点不屑于用插件来完成,自己写向行V ...