併發concurrent---3

来源:https://www.cnblogs.com/taojietaoge/archive/2019/01/22/10301711.html
-Advertisement-
Play Games

背景:併發知識是一個程式員段位升級的體現,同樣也是進入BAT的必經之路,有必要把併發知識重新梳理一遍。 ConcurrentHashMap:在有了併發的基礎知識以後,再來研究concurrent包。普通的HashMap為非線程安全的,在高併發場景下要使用線程安全版本的ConcurrentHashMa ...


背景:併發知識是一個程式員段位升級的體現,同樣也是進入BAT的必經之路,有必要把併發知識重新梳理一遍。

ConcurrentHashMap
在有了併發的基礎知識以後,再來研究concurrent包。普通的HashMap為非線程安全的,在高併發場景下要使用線程安全版本的ConcurrentHashMap;

眾所周知HashTable可以保證線程安全但卻效率低下,而HashMap是非線程安全但效率卻高於HashTable,於是ConcurrentHashMap就孕育而生成為二者的結合體,為了更好的理解ConcurrentHashMap先看下這兩個Map。

HashMap

HashMap之所以具有很快的訪問速度,因為它是根據鍵的hashCode值來存儲數據,在大多數情況下可以直接定位到它的值,但遍歷的順序是不確定的;HashMap的key可以為null,但是最多只允許一條記錄的鍵為null,另外允許多條記錄(value)的值為null,key為null的鍵值對永遠都放在一table[0]為頭節點的鏈表中;HashMap為非線程安全的,適用於單線程環境下,即在任一時刻都可以有多個線程同時對HashMap進行讀或寫操作,可以會導致數據的不一致;如果一定要使用HashMap又要保證線程安全,則可以用Collection的synchronizedMap方法或ConcurrentHashMap都OK;HashMap是基於哈希實表實現的,每一個元素是一個key-value對,其內部通過單鏈表結局衝突問題的,當Map容量不足(超過了閥值)時鏈表會自動增長;HashMap實現了Serializable介面,因此其支持序列化,並且實現了Cloneable介面,可以被克隆;

HashMap存儲數據的過程:

HashMap內部維護了一個存儲數據的Entry數組,HashMap採用鏈表解決衝突,每一個Entry本質上其實是一個單向鏈表;當要添加一個key-value對時,首先會通過hash(key)方法技術hash值,然後通過indexFor(hash,length)求該key-value對的存儲位置,其計算方法是先用Hash&0x7FFFFFFF後,再對length取模,這就保證了每一個key-value對都能存入HashMap,當計算出相同的位置是,由於存入位置是一個鏈表,所以把這個key-value對插入鏈表頭。

如上圖1 所示,最左邊豎列排的多個方格就代表哈希表,也叫哈希數組,數組的每個元素都是一個單鏈表的頭節點,鏈表是用來解決衝突的,如果不同的key映射到了數組的同一位置處,就將其放入單鏈表中;HashMap記憶體儲數據的Entry數組預設是16,如果沒有對Entry擴容機制的話,當存儲的數據一多,Entry內部的鏈表會很長,這就失去了HashMap的存儲意義了,所以HasnMap內部有自己的擴容機制(當size大於threshold時,對HashMap進行擴容)。 上圖2 是HashMap的鏈表存儲結構,其中E*代表一個Node節點,每個Node節點就對應著一個key-value的mapping映射;每個Node除了保存了key和value的映射之外,還保存了它下一Node的引用(Eb保存了Ebb的引用,而Ebb保存了Ebbb的引用);圖2中,每一個鏈表如Ec-->Ecc-->Eccc,這三個節點的key是不相等的。

分析HashMap源碼會發現其內部有幾個重要的變數如:size用於記錄HashMap的底層數組中已用槽的數量、threshold用於HashMap的閾值判斷,看是否需要調整HashMap的容量(threshold = 容量*載入因數)、DEFAULT_LOAD_FACTOR = 0.75f,即載入因數預設0.75。 HashMap的擴容是是新建了一個HashMap的底層數組,通過調用transfer方法,將就HashMap的全部元素添加到新的HashMap中(此步需要重新計算元素在新的數組中的索引位置,導致HashMap擴容成為一個相當耗時的操作),So我們在用HashMap的時,最好能提前預估下HashMap中元素的個數,這樣有助於提高HashMap的性能

HashTable
HashMap的功能與HashMap類似,如同樣是基於哈希表實現的、內部也是通過單鏈表解決衝突問題、容量不足時也會自動增加、同樣實現了Seriablizable介面支持序列化、實現了Cloneable介面可克隆;不同的是HashTable繼承自Dictionary類且為線程安全的(任一時間只有一個線程可以寫HashTable,但性能不如

ConcurrentHashMap),而HashMap繼承AbstractMap類且非線程安全。

如圖3,HashTable只有一把鎖,當一個線程訪問HashTable的同步方法時,會將整張table 鎖住,當其他線程也想訪問HashTable 同步方法時,就會進入阻塞或輪詢狀態。也就是確保同一時間只有一個線程對同步方法的占用,避免多個線程同時對數據的修改,由此確保線程的安全性;但HashTable 對get,put,remove 方法都使用了同步操作,這就造成如果兩個線程都只想使用get 方法去讀取數據時,因為一個線程先到進行了鎖操作,另一個線程就不得不等待,這樣必然導致效率低下,而且競爭越激烈,效率越低下。

ConcurrentHashMap(併發且線程安全)

ConcourrentHashMap是通過分段鎖技術來保證線程安全的[case:一個人到酒店開房可直接在前臺辦理入住,三個陌生人到酒店開房登記入住,另外兩個則要先排隊等第一個辦理結束(普通的Map),要是三個人所住的每個樓層都有一個可以辦理入住的前臺就無需排隊了(ConcurrentHashMap)];ConcurrentHashMap主要由Segment(桶)和HashEntry(節點)兩大數據組成,如下圖:

在hashMap 的基礎上,ConcurrentHashMap將數據分為多個segment(預設16個),然後每次操作對一個segment 加鎖,HashTable 在競爭激烈的併發環境下表現出效率低下的原因是由於所有訪問HashTable的線程都必須競爭同一把鎖,而ConcurrentHashMap將數據分到多個segment 中(預設16,也可在申明時自己設置,不過一旦設定就不能更改,擴容都是擴充各個segment 的容量),由於每個segment 都有一個自己的鎖,只要多個線程訪問的不是同一個segment 就沒有鎖爭用,就沒有堵塞,也就是允許16個線程併發的更新並且不存在鎖爭用現象。除此之外,ConcurrentHashMap的segment就類似一個HashTable,但比HashTable又更進一步優化,因為HashTable對get,put,remove方法都會使用鎖,而ConcurrnetHashMap中get方法是不涉及到鎖的;並且ConcurrentHashMap內部在併發讀取時,除了key 對應的value為null的情況下會用到鎖,其它的場景下都沒有用到鎖,所以對於讀操作無論多少線程併發都是安全高效的。

 




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

-Advertisement-
Play Games
更多相關文章
  • 在jquery中,給元素綁定事件,本文一共介紹三種方法,運用案例,針對最常用的on()方法,進行事件綁定操作。 事件綁定方法: ①$(element).bind() 參數:{ “事件名稱1”:function( ){ } ,“事件名稱2”:function(){ },......} 例如給類名為on ...
  • 一、23種設計模式分類: 二、設計模式的六大原則: 1、開閉原則(Open Close Principle):對擴展開放,對修改關閉。在程式需要進行拓展的時候,不能去修改原有的代碼,實現一個熱插拔的效果。 2、里氏代換原則(Liskov Substitution Principle):任何基類可以出 ...
  • DDD早於微服務「出道」十年,這兩個「忘年交」的軟體設計哲學是如何相愛相殺的? 背景 微服務現在可以說是軟體研發領域無人不提的話題,然而業界流行的對比多數都是所謂的Monolithic(單體應用),而大量的系統在十幾年前都已經是以SOA(面向服務架構)為基礎的分散式系統了,那麼微服務作為新的架構標準 ...
  • 靜兒就職的是新美大的基礎架構部門,做的是基於k8s的容器調度開發。k8s只是個工具,真正技術的上就是和網路打交道要多些。需要對網路中的數據流向有清晰的認識。大多數時間還是在做平臺,事情很跟業務很相似,也主要是工程。 之前記得有同事說過,設計的東西都被框架封裝好了,設計模式基本用不上。但是靜兒怎麼覺得 ...
  • 程式的三種基本結構 複合語句 將若幹個C語句使用花括弧{ }包括起來形成複合語句。花括弧內可以包含任何C語句, 其一般形式為: { 語句1; 語句2; …… 語句n; } 1.if條件分支語句 if語句有三種語法形式,構成三種分支結構。 1)流程(單選控制結構) 語句形式如下: if (表達式)語句 ...
  • 關於博客訪問量的問題,影響因素有很多,例如你的權重,你的博客數量,包括你的標題是否吸引人都是一個衡量的標準。 這些東西需要的是日積月累,今天我們從其中的一個維度入手:發帖時間。相信大家都明白,不論是csdn,博客園這種技術博客 還是今日頭條百度貼吧或者抖音快手這種娛樂論壇,都有自己的線上高峰期。例如 ...
  • 大概的樣子 這是大致的樣子~ 寫之前想說的 思路 這裡有一個問題,就是在使用ssh的時候會有一個等待用戶輸入密碼的過程。不能直接一條命令鏈接ssh。我們可以通過 expect 來解決。 不懂的可以百度一下,很簡單的。O(∩_∩)O哈哈~ 開始寫 首先 首先解決ssh等待輸入密碼的問題,首先 我們要安 ...
  • 1. 虎嗅網文章數據 寫在前面 今天繼續使用 爬取數據,很不幸,虎嗅資訊網被我選中了,網址為 爬的就是它的資訊頻道,本文章僅供學習交流使用,切勿用作其他用途。 常規操作,分析待爬取的頁面 拖拽頁面到最底部,會發現一個 按鈕,點擊之後,抓取一下請求,得到如下地址 2. 虎嗅網文章數據 分析請求 查閱該 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...