上一节我们说到了基本原子类的简单介绍,这一节我们先来看一下基本类型: 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. "You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near ''zhaoqiuyu' (`NAME`,`PRICE`,`COUNT`) values('电脑',1999,1)' at lin

    "You have an error in your SQL syntax; check the manual that corresponds to your MySQL server v ...

  2. C++中 / 和 % 在分离各位时的妙用

    在学习c++的过程中,我们一般用 / 和 % 来分解数字的各个位 取整 (/) 比如1234 / 10 等于 123.4,这相当于把前三位分解出来了 取余(%) 比如 12345 的分解方法 个位:1 ...

  3. Oracle数据库----查询

    --笛卡尔集select empno,ename, 员工表.deptno, 部门表.deptno, dname from 部门表, 员工表; --添加合适的条件,可以避免笛卡尔集,从而得到正确的多表查 ...

  4. zookeeper的客户端应用

    什么zookeeper? ZooKeeper是一个分布式的,开放源码的分布式应用程序协调服务,是Google的Chubby一个开源的实现,是Hadoop和Hbase的重要组件.它是一个为分布式应用提供 ...

  5. Java项目案例之---计算公司员工的工资(面向对象复习)

    计算公司员工的工资(面向对象复习) 某公司的雇员分为以下若干类: Employee:这是所有员工总的父类,属性:员工的姓名,员工的生日月份.方法:double getSalary(int month) ...

  6. Java基础之回味finally

    平时大家try…catch…finally语句用的不少,知道finally块一定会在try…catch..执行结束时执行,但是具体是在什么时候执行呢,今天我们一起来看下. public static ...

  7. 常见Code Review过程中发现的问题-续

    上一篇列举了一些比较常见的Code Review问题列表,文末有链接,可追溯查看.本篇为上篇的姊妹篇,继续列举一些上篇遗漏的或不易发现的问题清单,希望能整体性把一些常见的问题表述出来. 测试数据不具有 ...

  8. .net持续集成cake篇之cake任务依赖、自定义配置荐及环境变量读取

    系列目录 新建一个构建任务及任务依赖关系设置 上节我们通过新建一个HelloWorld示例讲解了如何编写build.cake以及如何下载build.ps1启动文件以及如何运行.实际项目中,我们使用最多 ...

  9. Mac上pycharm集成pyspark

    前提: 1.已经安装好spark.我的是spark2.2.0. 2.已经有python环境,我这边使用的是python3.6. 一.安装py4j 使用pip,运行如下命令: pip install p ...

  10. 自定义new和delete

    #include "stdafx.h" #include <stdlib.h> #include <malloc.h> #include <iostr ...