除了使用 synchronized、Lock 加锁之外,Java 中还有很多不需要加锁就可以解决并发问题的工具类

一、原子工具类

JDK 1.8 中,java.util.concurrent.atomic 包下类都是原子类,原子类都是基于 sun.misc.Unsafe 实现的。

  • CPU 为了解决并发问题,提供了 CAS 指令,全称 Compare And Swap,即比较并交互
  • CAS 指令需要 3 个参数,变量、比较值、新值。当变量的当前值与比较值相等时,才把变量更新为新值
  • CAS 是一条 CPU 指令,由 CPU 硬件级别上保证原子性
  • java.util.concurrent.atomic 包中的原子分为:原子性基本数据类型、原子性对象引用类型、原子性数组、原子性对象属性更新器和原子性累加器

原子性基本数据类型:AtomicBoolean、AtomicInteger、AtomicLong

原子性对象引用类型:AtomicReference、AtomicStampedReference、AtomicMarkableReference

原子性数组:AtomicIntegerArray、AtomicLongArray、AtomicReferenceArray

原子性对象属性更新:AtomicIntegerFieldUpdater、AtomicLongFieldUpdater、AtomicReferenceFieldUpdater

原子性累加器:DoubleAccumulator、DoubleAdder、LongAccumulator、LongAdder

修改我们之前测试原子性问题的类,使用 AtomicInteger 的简单例子

package constxiong.concurrency.a026;import java.util.concurrent.atomic.AtomicInteger;/*** 测试 原子类 AtomicInteger* * @author ConstXiong*/
public class TestAtomicInteger {// 计数变量static volatile AtomicInteger count = new AtomicInteger(0);public static void main(String[] args) throws InterruptedException {// 线程 1 给 count 加 10000Thread t1 = new Thread(() -> {for (int j = 0; j <10000; j++) {count.incrementAndGet();}System.out.println("thread t1 count 加 10000 结束");});// 线程 2 给 count 加 10000Thread t2 = new Thread(() -> {for (int j = 0; j <10000; j++) {count.incrementAndGet();}System.out.println("thread t2 count 加 10000 结束");});// 启动线程 1t1.start();// 启动线程 2t2.start();// 等待线程 1 执行完成t1.join();// 等待线程 2 执行完成t2.join();// 打印 count 变量System.out.println(count.get());}}

打印结果如预期

thread t2 count 加 10000 结束
thread t1 count 加 10000 结束
20000

二、线程本地存储

  • java.lang.ThreadLocal 类用于线程本地化存储。
  • 线程本地化存储,就是为每一个线程创建一个变量,只有本线程可以在该变量中查看和修改值。
  • 典型的使用例子就是,spring 在处理数据库事务问题的时候,就用了 ThreadLocal 为每个线程存储了各自的数据库连接 Connection。
  • 使用 ThreadLocal 要注意,在不使用该变量的时候,一定要调用 remove() 方法移除变量,否则可能造成内存泄漏的问题。

示例

package constxiong.concurrency.a026;/*** 测试 原子类 AtomicInteger* * @author ConstXiong*/
public class TestThreadLocal {// 线程本地存储变量private static final ThreadLocal<Integer> THREAD_LOCAL_NUM = new ThreadLocal<Integer>() {@Overrideprotected Integer initialValue() {//初始值return 0;}};public static void main(String[] args) {for (int i = 0; i <3; i++) {// 启动三个线程Thread t = new Thread() {@Overridepublic void run() {add10ByThreadLocal();}};t.start();}}/*** 线程本地存储变量加 5*/private static void add10ByThreadLocal() {try {for (int i = 0; i <5; i++) {Integer n = THREAD_LOCAL_NUM.get();n += 1;THREAD_LOCAL_NUM.set(n);System.out.println(Thread.currentThread().getName() + " : ThreadLocal num=" + n);}} finally {THREAD_LOCAL_NUM.remove();// 将变量移除}}
}

每个线程最后一个值都打印到了 5

Thread-0 : ThreadLocal num=1
Thread-2 : ThreadLocal num=1
Thread-1 : ThreadLocal num=1
Thread-2 : ThreadLocal num=2
Thread-0 : ThreadLocal num=2
Thread-2 : ThreadLocal num=3
Thread-0 : ThreadLocal num=3
Thread-1 : ThreadLocal num=2
Thread-0 : ThreadLocal num=4
Thread-2 : ThreadLocal num=4
Thread-0 : ThreadLocal num=5
Thread-1 : ThreadLocal num=3
Thread-2 : ThreadLocal num=5
Thread-1 : ThreadLocal num=4
Thread-1 : ThreadLocal num=5

三、copy-on-write

  • 根据英文名称可以看出,需要写时复制,体现的是一种延时策略。
  • Java 中的 copy-on-write 容器包括:CopyOnWriteArrayList、CopyOnWriteArraySet。
  • 涉及到数组的全量复制,所以也比较耗内存,在写少的情况下使用比较适合。

简单的 CopyOnWriteArrayList 的示例,这里只是说明 CopyOnWriteArrayList 怎么用,并且是线程安全的。这个场景并不适合使用 CopyOnWriteArrayList,因为写多读少

package constxiong.concurrency.a026;import java.util.ArrayList;
import java.util.List;
import java.util.Random;
import java.util.concurrent.CopyOnWriteArrayList;/*** 测试 copy-on-write* @author ConstXiong*/
public class TestCopyOnWrite {private static final Random R = new Random();private static CopyOnWriteArrayList<Integer> cowList = new CopyOnWriteArrayList<Integer>();
// private static ArrayList<Integer> cowList = new ArrayList<Integer>();public static void main(String[] args) throws InterruptedException {List<Thread> threadList = new ArrayList<Thread>();//启动 1000 个线程,向 cowList 添加 5 个随机整数for (int i = 0; i <1000; i++) {Thread t = new Thread(() -> {for (int j = 0; j <5; j++) {//休眠 10 毫秒,让线程同时向 cowList 添加整数,引出并发问题try {Thread.sleep(10);} catch (InterruptedException e) {e.printStackTrace();}cowList.add(R.nextInt(100));}}) ;t.start();threadList.add(t);}for (Thread t : threadList) {t.join();}System.out.println(cowList.size());}
}

打印结果

5000

如果把

private static CopyOnWriteArrayList<Integer> cowList = new CopyOnWriteArrayList<Integer>();

改为

private static ArrayList<Integer> cowList = new ArrayList<Integer>();

打印结果就是小于 5000 的整数了

Java中有哪些无锁技术来解决并发问题?如何使用?相关推荐

  1. Java CAS无锁技术深度解析

    在看AQS代码的时候,看到它很多地方调用了unsafe.compareAndSwapInt这类方法,百度了一下得知这叫CAS无锁技术. CAS原理深度分析 转自:https://blog.csdn.n ...

  2. java轻松实现无锁队列

    1.什么是无锁(Lock-Free)编程 当谈及 Lock-Free 编程时,我们常将其概念与 Mutex(互斥) 或 Lock(锁) 联系在一起,描述要在编程中尽量少使用这些锁结构,降低线程间互相阻 ...

  3. 掌握Java的内存模型,你就是解决并发问题最靓的仔

    摘要:如果编写的并发程序出现问题时,很难通过调试来解决相应的问题,此时,需要一行行的检查代码,这个时候,如果充分理解并掌握了Java的内存模型,你就能够很快分析并定位出问题所在. 本文分享自华为云社区 ...

  4. java中有那三种技术平台_Java语言有三种技术平台,分别是JavaSE、JavaME、JavaEE()...

    种技唐朝爱情诗歌丰富多彩的原因是( ). 术平大脑皮层主要运动区特点:(). 分别指导合成真核生物蛋白质的序列主要是 全部属于发声共鸣腔的是().A.种技声带.种技口腔.牙齿.口唇:B.咽喉.口腔.牙 ...

  5. 实战并发编程 - 02解决并发问题常用套路

    文章目录 Pre 解决并发问题的方法 无锁的方式解决并发问题 局部变量 不可变对象 ThreadLocal CAS原子类 有锁的方式解决并发问题 ReentrantLock可重入锁 synchroni ...

  6. java 高并发_Java 高并发之无锁(CAS)

    Java 高并发之无锁(CAS) 本篇主要讲 Java中的无锁 CAS ,无锁 顾名思义就是 以不上锁的方式解决并发问题,而不使用synchronized 和 lock 等.. 1. Atomic 包 ...

  7. cas无法使用_【漫画】CAS原理分析!无锁原子类也能解决并发问题!

    本文来源于微信公众号[胖滚猪学编程].转载请注明出处 在漫画并发编程系统博文中,我们讲了N篇关于锁的知识,确实,锁是解决并发问题的万能钥匙,可是并发问题只有锁能解决吗?今天要出场一个大BOSS:CAS ...

  8. C语言链表无锁化CAS,CAS无锁操作

    主要讲的是<Implementing Lock-Free Queues>的论点,具体直接看论文最好.这里总结些要点. CAS就是Compare And Swap.gcc可以调用: __sy ...

  9. 无锁队列的几种实现及其性能对比

    一.无锁队列用在什么样的场景? 当需要处理的数据非常多,比如行情数据,一秒处理非常多的数据的时候,可以考虑用无锁队列.但是如果一秒只需要处理几百或者几千的数据,是没有必要考虑用无锁队列的.用互斥锁就能 ...

最新文章

  1. C++中函数指针的使用
  2. JQuery 选择器。
  3. 案例式c语言程序设计陈明,C语言程序设计教程 清华大学出版社 陈明主编 第1章答案...
  4. [原创] 盟军敢死队2 - 3D模型/动作浏览器
  5. c 解析java byte,深入解析Java编程中面向字节流的一些应用
  6. Typecho添加投稿功能插件
  7. oracle net conf启动无反应,weblogic突然无法启动,显示Server state changed to FORCE
  8. windows 弹出 api-ms-win-crt-runtime-l1-1-0.dll 丢失的问题
  9. json数组转java集合
  10. 程序设计题大病:劝你不要买未来教育的Python二级书
  11. 绕过校园网认证实现免费上网【三端】
  12. JAVA 调用 labview_制作软接入点ESP8266并通过labview读取数据
  13. 【计算理论】图灵机 ( 图灵机设计 )
  14. 王石:中国楼市泡沫必破 目前只是时间问题
  15. 大学生程序设计邀请赛(华东师范大学)A
  16. 2021年中国服装行业经营现状及重点企业对比分析[图]
  17. 群晖synology LDAP server 同步
  18. trans系列平移距离模型
  19. python unzip解压缩_Python压缩和解压缩文件(zip/unzip)
  20. MEM/MBA数学基础(08)数据分析

热门文章

  1. Ubuntu安装smplayer播放器
  2. 2015年百度二面试题
  3. 信息系统项目管理师:论项目的质量管理
  4. 基于NHibernate的UnitOfWork+Repository模式(AutoFac)–Part2
  5. Gartner:大数据投资增长,但计划投资的组织机构却在减少
  6. ThreadLocal_OSIV模式_FIlter_Web ajax
  7. js 获取表格数据(表单变量值)
  8. SQL Server 2012 Express LocalDB
  9. solaris vi 方向键问题
  10. WebSocket+MSE——HTML5 直播技术解析