亚洲视频二区_亚洲欧洲日本天天堂在线观看_日韩一区二区在线观看_中文字幕不卡一区

公告:魔扣目錄網(wǎng)為廣大站長提供免費(fèi)收錄網(wǎng)站服務(wù),提交前請做好本站友鏈:【 網(wǎng)站目錄:http://www.430618.com 】, 免友鏈快審服務(wù)(50元/站),

點(diǎn)擊這里在線咨詢客服
新站提交
  • 網(wǎng)站:51998
  • 待審:31
  • 小程序:12
  • 文章:1030137
  • 會員:747


Java中有哪些無鎖技術(shù)來解決并發(fā)問題?如何使用?

 

除了使用 synchronized、Lock 加鎖之外,JAVA 中還有很多不需要加鎖就可以解決并發(fā)問題的工具類

一、原子工具類

JDK 1.8 中,java.util.concurrent.atomic 包下類都是原子類,原子類都是基于 sun.misc.Unsafe 實(shí)現(xiàn)的。

  • CPU 為了解決并發(fā)問題,提供了 CAS 指令,全稱 Compare And Swap,即比較并交互
  • CAS 指令需要 3 個參數(shù),變量、比較值、新值。當(dāng)變量的當(dāng)前值與比較值相等時,才把變量更新為新值
  • CAS 是一條 CPU 指令,由 CPU 硬件級別上保證原子性
  • java.util.concurrent.atomic 包中的原子分為:原子性基本數(shù)據(jù)類型、原子性對象引用類型、原子性數(shù)組、原子性對象屬性更新器和原子性累加器

原子性基本數(shù)據(jù)類型:AtomicBoolean、AtomicInteger、AtomicLong

原子性對象引用類型:AtomicReference、AtomicStampedReference、AtomicMarkableReference

原子性數(shù)組: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 {

 // 計(jì)數(shù)變量
 static volatile AtomicInteger count = new AtomicInteger(0);

 public static void main(String[] args) throws InterruptedException {
 // 線程 1 給 count 加 10000
 Thread t1 = new Thread(() -> {
 for (int j = 0; j <10000; j++) {
 count.incrementAndGet();
 }
 System.out.println("thread t1 count 加 10000 結(jié)束");
 });

 // 線程 2 給 count 加 10000
 Thread t2 = new Thread(() -> {
 for (int j = 0; j <10000; j++) {
 count.incrementAndGet();
 }
 System.out.println("thread t2 count 加 10000 結(jié)束");
 });

 // 啟動線程 1
 t1.start();
 // 啟動線程 2
 t2.start();

 // 等待線程 1 執(zhí)行完成
 t1.join();
 // 等待線程 2 執(zhí)行完成
 t2.join();

 // 打印 count 變量
 System.out.println(count.get());
 }

}

打印結(jié)果如預(yù)期

thread t2 count 加 10000 結(jié)束
thread t1 count 加 10000 結(jié)束
20000
Java中有哪些無鎖技術(shù)來解決并發(fā)問題?如何使用?

 

二、線程本地存儲

  • java.lang.ThreadLocal 類用于線程本地化存儲。
  • 線程本地化存儲,就是為每一個線程創(chuàng)建一個變量,只有本線程可以在該變量中查看和修改值。
  • 典型的使用例子就是,spring 在處理數(shù)據(jù)庫事務(wù)問題的時候,就用了 ThreadLocal 為每個線程存儲了各自的數(shù)據(jù)庫連接 Connection。
  • 使用 ThreadLocal 要注意,在不使用該變量的時候,一定要調(diào)用 remove() 方法移除變量,否則可能造成內(nèi)存泄漏的問題。

示例

package constxiong.concurrency.a026;

/**
 * 測試 原子類 AtomicInteger
 * 
 * @author ConstXiong
 */
public class TestThreadLocal {

 // 線程本地存儲變量
 private static final ThreadLocal<Integer> THREAD_LOCAL_NUM = new ThreadLocal<Integer>() {
 @Override
 protected Integer initialValue() {//初始值
 return 0;
 }
 };

 public static void main(String[] args) {
 for (int i = 0; i <3; i++) {// 啟動三個線程
 Thread t = new Thread() {
 @Override
 public 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

 

Java中有哪些無鎖技術(shù)來解決并發(fā)問題?如何使用?

 

三、copy-on-write

  • 根據(jù)英文名稱可以看出,需要寫時復(fù)制,體現(xiàn)的是一種延時策略。
  • Java 中的 copy-on-write 容器包括:CopyOnWriteArrayList、CopyOnWriteArraySet。
  • 涉及到數(shù)組的全量復(fù)制,所以也比較耗內(nèi)存,在寫少的情況下使用比較適合。

簡單的 CopyOnWriteArrayList 的示例,這里只是說明 CopyOnWriteArrayList 怎么用,并且是線程安全的。這個場景并不適合使用 CopyOnWriteArrayList,因?yàn)閷懚嘧x少。

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 個隨機(jī)整數(shù)
 for (int i = 0; i <1000; i++) {
 Thread t = new Thread(() -> {
 for (int j = 0; j <5; j++) {
 //休眠 10 毫秒,讓線程同時向 cowList 添加整數(shù),引出并發(fā)問題
 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());
 }
}

打印結(jié)果

5000

如果把

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

改為

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

打印結(jié)果就是小于 5000 的整數(shù)了

分享到:
標(biāo)簽:技術(shù) Java
用戶無頭像

網(wǎng)友整理

注冊時間:

網(wǎng)站:5 個   小程序:0 個  文章:12 篇

  • 51998

    網(wǎng)站

  • 12

    小程序

  • 1030137

    文章

  • 747

    會員

趕快注冊賬號,推廣您的網(wǎng)站吧!
最新入駐小程序

數(shù)獨(dú)大挑戰(zhàn)2018-06-03

數(shù)獨(dú)一種數(shù)學(xué)游戲,玩家需要根據(jù)9

答題星2018-06-03

您可以通過答題星輕松地創(chuàng)建試卷

全階人生考試2018-06-03

各種考試題,題庫,初中,高中,大學(xué)四六

運(yùn)動步數(shù)有氧達(dá)人2018-06-03

記錄運(yùn)動步數(shù),積累氧氣值。還可偷

每日養(yǎng)生app2018-06-03

每日養(yǎng)生,天天健康

體育訓(xùn)練成績評定2018-06-03

通用課目體育訓(xùn)練成績評定