Java併發包中CopyOnWrite容器相關類簡介

来源:http://www.cnblogs.com/nullzx/archive/2017/08/30/7455919.html
-Advertisement-
Play Games

簡介: 本文是主要介紹,併發容器CopyOnWriteArrayList和CopyOnWriteArraySet(不含重覆元素的併發容器)的基本原理和使用示例。 歡迎探討,如有錯誤敬請指正 如需轉載,請註明出處 http://www.cnblogs.com/nullzx/ 1. CopyOnWrit ...


簡介

本文是主要介紹,併發容器CopyOnWriteArrayList和CopyOnWriteArraySet(不含重覆元素的併發容器)的基本原理和使用示例。

歡迎探討,如有錯誤敬請指正

如需轉載,請註明出處 http://www.cnblogs.com/nullzx/


1. CopyOnWriteArrayList

從類的名字我們可以看出,該類是基於ArrayList類實現的。而CopyOnWrite的意思顯然借鑒了操作系統中寫時拷貝的思想。該容器主要有以下特點:

1)讀取該容器中元素時,不加鎖。

2)寫操作,會加鎖,也就是說多個線程進行寫入操作時會逐個獲取鎖後進行寫入。

3)寫操作不會影響讀操作,也就是說線程要進行讀操作時,不會因為有線程正在進行寫操作而阻塞。

下麵是寫操作源代碼

    public E set(int index, E element) {
        final ReentrantLock lock = this.lock;
        lock.lock();
        try {
            Object[] elements = getArray();
            E oldValue = get(elements, index);

            if (oldValue != element) {
                int len = elements.length;
                Object[] newElements = Arrays.copyOf(elements, len);
                newElements[index] = element;
                setArray(newElements);
            } else {
                // Not quite a no-op; ensures volatile write semantics
                setArray(elements);
            }
            return oldValue;
        } finally {
            lock.unlock();
        }
    }

工作原理:在CopyOnWriteArrayList類中,定一了一個數組private transient volatile Object[] array;容器中存儲的對象的索引都會放在這個數組中。

寫操作首先會獲取鎖。當獲取鎖成功後,複製該數組到一個新的數組newElements中,然後修改或者添加某個元素(註意這個時候如果有線程來讀取該數組中的某個值,由於讀操作不需要獲取鎖,所以不會被阻塞,但是可能不能讀取到最新修改後的值)。修改後,讓array指向經過修改後的新數組newElements,原array指向的數組會被垃圾回收器回收。

 

下麵的代碼是CopyOnWriteArrayList的演示常式

package javalearning;

import java.util.Random;
import java.util.concurrent.CopyOnWriteArrayList;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;


/*CopyOnWriteArrayList演示常式*/
public class CopyOnWriteArrayListDemo {
	/*定義一個CopyOnWriteArrayList對象,讀線程和寫線程會同時使用它*/
	private CopyOnWriteArrayList<Integer> cowal = new CopyOnWriteArrayList<Integer>();
	{
		/*對CopyOnWriteArrayList對象初始化*/
		cowal.add(1);
		cowal.add(2);
		cowal.add(3);
	}
	
	private Random rnd = new Random();
	
	public class ReadThread implements Runnable{
		private String id;
		
		public ReadThread(String id){
			this.id = id;
		}
		
		@Override
		public void run() {
			try {
				Thread.sleep(rnd.nextInt(1000));
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
			/*讀線程會列印出CopyOnWriteArrayList對象的所有數值*/
			System.out.println(id + " " + cowal.toString());
		}
		
	}
	
	public class WriteThread implements Runnable{
		
		/*寫線程會將CopyOnWriteArrayList對象的所有數值加1*/
		@Override
		public void run() {
			for(int i = 0; i < cowal.size(); i++){
				int x = cowal.get(i);
				
				/*每修改一個元素之前加鎖*/
				cowal.set(i, x+1);
				/*修改完一個元素後釋放鎖*/
				
				try{
					Thread.sleep(rnd.nextInt(1000));
				}catch(InterruptedException e){
					e.printStackTrace();
				}
			}
		}
		
	}
	
	public static void main(String[] args){
		ExecutorService es = Executors.newCachedThreadPool();
		CopyOnWriteArrayListDemo demo = new CopyOnWriteArrayListDemo();
		/*創建兩個讀線程*/
		es.execute(demo.new ReadThread("r1"));
		es.execute(demo.new ReadThread("r2"));
		/*創建兩個寫線程*/
		es.execute(demo.new WriteThread());
		es.execute(demo.new WriteThread());
		
		es.shutdown();
		while(!es.isTerminated()){
			;
		}
		System.out.println("=====================");
		/*CopyOnWriteArrayList對象中的最終值*/
		System.out.println("eventual " + demo.cowal.toString());
	}
}

全部結束後,最終結果和我們預想的一致。

r2 [2, 3, 3]
r1 [2, 4, 3]
=====================
eventual [2, 4, 5]

2. CopyOnWriteArraySet

CopyOnWriteArraySet是一個不存貯重覆對象的寫時拷貝容器。它的實現的原理很簡單,在其內部定義了一個CopyOnWriteArrayList對象al,當向該容器添加一個對象時,會調用addIfAbsent方法。

    public boolean add(E e) {
        return al.addIfAbsent(e);
    }

3. 參考內容

[1] 聊聊併發-Java中的Copy-On-Write容器 | 併發編程網 – ifeve.com

[2] Java併發編程:併發容器之CopyOnWriteArrayList(轉載)


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • C# using 三種使用方式 http://www.cnblogs.com/dachengxiaomeng/p/7452021.html 1.using指令。 using 命名空間名字。例如: using System; 這樣可以在程式中直接用命令空間中的類型,而不必指定類型的詳細命名空間,類似於 ...
  • jQuery的方法連綴使用起來非常方便,可以簡化語句,讓代碼變得清晰簡潔。那C#的類方法能不能也實現類似的功能呢?基於這樣的疑惑,研究了一下jQuery的源代碼,發現就是需要方法連綴的函數方法最後返回對象本身即可。既然javascript可以,C#應該也是可以的。 為了驗證,編寫一個jQPerson ...
  • 最近在做項目進度管理時,想通過安裝net.sf.mpxj-for-csharp包讀取.mpp格式文件,通過Nuget線上安裝時,出現以下情況,無法安裝,故開啟離線安裝道路。 離線安裝步驟如下: 一、下載你需要安裝的離線包nupkg文件,可以在Nuget官網下載:https://www.nuget.o ...
  • 最近在開始一個微信開發,發現微信的Access_Token獲取每天次數是有限的,然後想到緩存,正好看到微信教程裡面推薦HttpRuntime.Cache緩存就順便看了下。 ...
  • //設置對話框的過濾條件 ofdSelectPic.Filter = "png文件(*.png)|*.png|jpg 文件(*.jpg)|*.jpg|所有文件(*.*)|*.*"; ofdSelectPic.Title = "打開圖片"; ofdSelectPic.FilterIndex = 2; ...
  • 在使用Entity Framework過程中,有時需要藉助工具生成Code First的代碼,而Entity Framework Reverse POCO Code First Generator是一款不錯的工具 在Visual Studio中,通過“工具”→“擴展和更新...”來安裝Entity ...
  • 游戲伺服器裡面總是有一大堆的配置文件需要讀取, 而且這些配置文件的讀取: * 要不然做成弱類型的, 就是一堆字元串或者數字, 不能看出來錯誤(需要重新檢測一次) * 要不然做成強類型的, 每種類型都需要自己Parse一下 我個人比較喜歡後者, 因為前者LoadConfig的代碼簡單, 但是寫邏輯的時 ...
  • Django預設使用的文件存儲系統'django.core.files.storage.FileSystemStorage'是一個本地存儲系統,由settings中的DEFAULT_FILE_STORAGE值確定。 class FileSystemStorage(location=None, bas ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...