亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Java的CopyOnWriteArrayList操作詳解

 更新時(shí)間:2023年12月21日 08:46:01   作者:weixin_33979363  
這篇文章主要介紹了Java的CopyOnWriteArrayList操作詳解,  CopyOnWriteArrayList是ArrayList 的一個(gè)線(xiàn)程安全的變體,其中所有可變操作(add、set等等)都是通過(guò)對(duì)底層數(shù)組進(jìn)行一次新的復(fù)制來(lái)實(shí)現(xiàn)的,需要的朋友可以參考下

CopyOnWriteArrayList

CopyOnWriteArrayList是ArrayList 的一個(gè)線(xiàn)程安全的變體,其中所有可變操作(add、set等等)都是通過(guò)對(duì)底層數(shù)組進(jìn)行一次新的復(fù)制來(lái)實(shí)現(xiàn)的。

這一般需要很大的開(kāi)銷(xiāo),但是當(dāng)遍歷操作的數(shù)量大大超過(guò)可變操作的數(shù)量時(shí),這種方法可能比其他替代方法更 有效。在不能或不想進(jìn)行同步遍歷,但又需要從并發(fā)線(xiàn)程中排除沖突時(shí),它也很有用。“快照”風(fēng)格的迭代器方法在創(chuàng)建迭代器時(shí)使用了對(duì)數(shù)組狀態(tài)的引用。此數(shù)組在迭代器的生存期內(nèi)不會(huì)更改,因此不可能發(fā)生沖突,并且迭代器保證不會(huì)拋出ConcurrentModificationException。創(chuàng)建迭代器以后,迭代器就不會(huì)反映列表的添加、移除或者更改。在迭代器上進(jìn)行的元素更改操作(remove、set和add)不受支持。這些方法將拋出UnsupportedOperationException。允許使用所有元素,包括null。

內(nèi)存一致性效果:當(dāng)存在其他并發(fā) collection 時(shí),將對(duì)象放入CopyOnWriteArrayList之前的線(xiàn)程中的操作 happen-before 隨后通過(guò)另一線(xiàn)程從CopyOnWriteArrayList中訪(fǎng)問(wèn)或移除該元素的操作。

這種情況一般在多線(xiàn)程操作時(shí),一個(gè)線(xiàn)程對(duì)list進(jìn)行修改。一個(gè)線(xiàn)程對(duì)list進(jìn)行fore時(shí)會(huì)出現(xiàn)java.util.ConcurrentModificationException錯(cuò)誤。

下面來(lái)看一個(gè)列子:兩個(gè)線(xiàn)程一個(gè)線(xiàn)程fore一個(gè)線(xiàn)程修改list的值。

package com.lucky.concurrent.list;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
public class CopyOnWriteArrayListDemo {
	/**
	 * 讀線(xiàn)程
	 * @author wangjie
	 *
	 */
	private static class ReadTask implements Runnable {
		List<String> list;
		public ReadTask(List<String> list) {
			this.list = list;
		}
		public void run() {
			for (String str : list) {
				System.out.println(str);
			}
		}
	}
	/**
	 * 寫(xiě)線(xiàn)程
	 * @author wangjie
	 *
	 */
	private static class WriteTask implements Runnable {
		List<String> list;
		int index;
		public WriteTask(List<String> list, int index) {
			this.list = list;
			this.index = index;
		}
		public void run() {
			list.remove(index);
			list.add(index, "write_" + index);
		}
	}
	public void run() {
		final int NUM = 10;
		List<String> list = new ArrayList<String>();
		for (int i = 0; i < NUM; i++) {
			list.add("main_" + i);
		}
		ExecutorService executorService = Executors.newFixedThreadPool(NUM);
		for (int i = 0; i < NUM; i++) {
			executorService.execute(new ReadTask(list));
			executorService.execute(new WriteTask(list, i));
		}
		executorService.shutdown();
	}
	public static void main(String[] args) {
		new CopyOnWriteArrayListDemo().run();
	}
}

從結(jié)果中可以看出來(lái)。在多線(xiàn)程情況下報(bào)錯(cuò)。其原因就是多線(xiàn)程操作結(jié)果:那這個(gè)種方案不行我們就換個(gè)方案。

用jdk自帶的類(lèi)CopyOnWriteArrayList來(lái)做容器。

這個(gè)類(lèi)和ArrayList最大的區(qū)別就是add(E) 的時(shí)候。容器會(huì)自動(dòng)copy一份出來(lái)然后再尾部add(E)。

看源碼:

/**
     * Appends the specified element to the end of this list.
     *
     * @param e element to be appended to this list
     * @return <tt>true</tt> (as specified by {@link Collection#add})
     */
    public boolean add(E e) {
	final ReentrantLock lock = this.lock;
	lock.lock();
	try {
	    Object[] elements = getArray();
	    int len = elements.length;
	    Object[] newElements = Arrays.copyOf(elements, len + 1);
	    newElements[len] = e;
	    setArray(newElements);
	    return true;
	} finally {
	    lock.unlock();
	}
    }

用到了Arrays.copyOf 方法。這樣導(dǎo)致每次操作的都不是同一個(gè)引用。也就不會(huì)出現(xiàn)java.util.ConcurrentModificationException錯(cuò)誤。

換了種方案看代碼:

//		List<String> list = new ArrayList<String>();
		CopyOnWriteArrayList<String> list = new CopyOnWriteArrayList<String>();

也就把容器list換成了 CopyOnWriteArrayList,其他的沒(méi)變。

線(xiàn)程里面的list不用改。因?yàn)?CopyOnWriteArrayList實(shí)現(xiàn)的也是list<E> 接口。

其結(jié)果沒(méi)報(bào)錯(cuò)。 CopyOnWriteArrayList add(E) 和remove(int index)都是對(duì)新的數(shù)組進(jìn)行修改和新增。

所以在多線(xiàn)程操作時(shí)不會(huì)出現(xiàn)java.util.ConcurrentModificationException錯(cuò)誤。

所以最后得出結(jié)論:CopyOnWriteArrayList適合使用在讀操作遠(yuǎn)遠(yuǎn)大于寫(xiě)操作的場(chǎng)景里,比如緩存。

發(fā)生修改時(shí)候做copy,新老版本分離,保證讀的高性能,適用于以讀為主的情況。

到此這篇關(guān)于Java的CopyOnWriteArrayList操作詳解的文章就介紹到這了,更多相關(guān)CopyOnWriteArrayList操作內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • spring-cloud-stream結(jié)合kafka使用詳解

    spring-cloud-stream結(jié)合kafka使用詳解

    這篇文章主要介紹了spring-cloud-stream結(jié)合kafka使用詳解,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-08-08
  • 貨拉拉大數(shù)據(jù)對(duì)BitMap的探索實(shí)踐詳解

    貨拉拉大數(shù)據(jù)對(duì)BitMap的探索實(shí)踐詳解

    這篇文章主要為大家介紹了貨拉拉大數(shù)據(jù)對(duì)BitMap的探索實(shí)踐詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-09-09
  • Spring boot + thymeleaf 后端直接給onclick函數(shù)賦值的實(shí)現(xiàn)代碼

    Spring boot + thymeleaf 后端直接給onclick函數(shù)賦值的實(shí)現(xiàn)代碼

    這篇文章主要介紹了Spring boot + thymeleaf 后端直接給onclick函數(shù)賦值的實(shí)現(xiàn)代碼,需要的朋友可以參考下
    2017-06-06
  • Java解決線(xiàn)程的不安全問(wèn)題之volatile關(guān)鍵字詳解

    Java解決線(xiàn)程的不安全問(wèn)題之volatile關(guān)鍵字詳解

    這篇文章主要介紹了Java解決線(xiàn)程的不安全問(wèn)題之volatile關(guān)鍵字詳解,可見(jiàn)性指一個(gè)線(xiàn)程對(duì)共享變量值的修改,能夠及時(shí)地被其他線(xiàn)程看到,而 volatile 關(guān)鍵字就保證內(nèi)存的可見(jiàn)性,需要的朋友可以參考下
    2023-08-08
  • SpringBoot請(qǐng)求處理之常用參數(shù)注解介紹與源碼分析

    SpringBoot請(qǐng)求處理之常用參數(shù)注解介紹與源碼分析

    SpringBoot是一種整合Spring技術(shù)棧的方式(或者說(shuō)是框架),同時(shí)也是簡(jiǎn)化Spring的一種快速開(kāi)發(fā)的腳手架,本篇讓我們一起學(xué)習(xí)請(qǐng)求處理、常用注解和方法參數(shù)的小技巧
    2022-10-10
  • java網(wǎng)上圖書(shū)商城(8)訂單模塊3

    java網(wǎng)上圖書(shū)商城(8)訂單模塊3

    這篇文章主要為大家詳細(xì)介紹了java網(wǎng)上圖書(shū)商城,訂單模塊第三篇,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2016-12-12
  • Spring 實(shí)現(xiàn)數(shù)據(jù)庫(kù)讀寫(xiě)分離的示例

    Spring 實(shí)現(xiàn)數(shù)據(jù)庫(kù)讀寫(xiě)分離的示例

    現(xiàn)在大型的電子商務(wù)系統(tǒng),在數(shù)據(jù)庫(kù)層面大都采用讀寫(xiě)分離技術(shù),我們通常的做法就是把查詢(xún)從主庫(kù)中抽取出來(lái),采用多個(gè)從庫(kù),使用負(fù)載均衡,減輕每個(gè)從庫(kù)的查詢(xún)壓力。
    2017-01-01
  • springboot 中文件上傳下載實(shí)例代碼

    springboot 中文件上傳下載實(shí)例代碼

    Spring Boot是由Pivotal團(tuán)隊(duì)提供的全新框架,其設(shè)計(jì)目的是用來(lái)簡(jiǎn)化新Spring應(yīng)用的初始搭建以及開(kāi)發(fā)過(guò)程。這篇文章主要介紹了springboot 中文件上傳下載實(shí)例代碼,需要的朋友可以參考下
    2017-11-11
  • Java中的FutureTask源碼解析

    Java中的FutureTask源碼解析

    這篇文章主要介紹了Java中的FutureTask源碼解析,FutureTask是一個(gè)可取消的異步計(jì)算,這個(gè)類(lèi)是Future的實(shí)現(xiàn)類(lèi),有開(kāi)始和取消一個(gè)計(jì)算的方法,如果一個(gè)計(jì)算已經(jīng)完成可以查看結(jié)果,需要的朋友可以參考下
    2023-12-12
  • Java多線(xiàn)程案例之定時(shí)器詳解

    Java多線(xiàn)程案例之定時(shí)器詳解

    定時(shí)器也是軟件開(kāi)發(fā)中的一個(gè)重要組件.?類(lèi)似于一個(gè)?“鬧鐘”.?達(dá)到一個(gè)設(shè)定的時(shí)間之后,?就執(zhí)行某個(gè)指定好的代碼。本文就來(lái)和大家詳細(xì)聊聊定時(shí)器的原理與使用,感興趣的可以了解一下
    2022-10-10

最新評(píng)論