日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程语言 > java >内容正文

java

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

發布時間:2025/3/21 java 15 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Java中有哪些无锁技术来解决并发问题?如何使用? 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

除了使用 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中有哪些无锁技术来解决并发问题?如何使用?的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

主站蜘蛛池模板: 神马久久久久 | 久久刺激| 99re6在线观看| a级片视频网站 | 91精品国产色综合久久不卡蜜臀 | 区一区二视频 | 亚洲欧美日韩中文字幕在线观看 | 黄色日韩在线 | 性做久久久久久久 | 男人天堂tv | 亚洲成人免费在线 | 免费在线观看成年人视频 | 久久99操 | 色先锋av资源 | 亚洲欧美综合色 | 自拍偷拍亚洲欧洲 | 日本精品一区视频 | 日本高清不卡码 | 无遮挡又爽又刺激的视频 | 高h免费视频 | 久久夜色精品国产欧美乱极品 | 欧美日韩xxxx | 国产成人综合在线观看 | 蜜桃免费av | 国产精品美女久久久 | 免费在线观看毛片 | 日本黄色免费在线观看 | 推特裸体gay猛交gay | 免费看黄色一级视频 | 99爱精品| 精品美女视频 | 国产高清在线不卡 | 色伊人av | 91直接进入 | 色六月婷婷 | av漫画在线观看 | 国产一线天粉嫩馒头极品av | 欧美午夜精品一区二区三区电影 | 国产在线精品福利 | 天天天天躁天天爱天天碰2018 | 一区二区av在线 | 黄色美女毛片 | 性欧美8khd高清极品 | 精品免费观看 | 成人在线国产精品 | 欧美精品中文 | 18禁免费无码无遮挡不卡网站 | 91香蕉视频在线看 | 中国av一区二区 | 伊人性视频 | 午夜国产一区二区三区 | 麻豆视频网址 | 亚洲成a人v欧美综合天堂麻豆 | 日日夜夜人人 | 国产精品一二三区在线观看 | 三级理论电影 | 香蕉色网 | 911香蕉视频| 久久发布国产伦子伦精品 | 日本美女a级片 | 中国一及毛片 | 在线看国产精品 | av综合久久 | 超碰人人在线 | 伊人福利视频 | 天天干天天干天天操 | 国产成人精品视频在线观看 | 中文在线天堂网 | www.欧美激情 | 视频成人免费 | 闫嫩的18sex少妇hd | 国产91嫩草 | 国产激情自拍 | 欧美精品久久久久久久多人混战 | 91免费在线| 久久久99精品国产一区二区三区 | 亚洲人在线视频 | 五月天激情四射 | 国产精品久久久久久久裸模 | 亚洲精品色图 | 制服一区二区 | 免费吃奶摸下激烈视频 | 精品久久免费观看 | 在线观看欧美亚洲 | 免费成人av在线播放 | 亚洲精品99 | 小柔好湿好紧太爽了国产网址 | 毛片a片免费看 | 亚洲欧美综合一区二区 | 西西午夜视频 | 欧美一区二区三区四区视频 | 国产草草草 | 在线青草 | 中文字幕一区二区人妻 | 97在线免费公开视频 | 三级男人添奶爽爽爽视频 | 一级片黄色片 | 日韩视频一区二区在线观看 | 亚洲视频免费播放 |