高性能隊列disruptor為什麼這麼快?

来源:https://www.cnblogs.com/lewis09/archive/2018/11/17/9974995.html
-Advertisement-
Play Games

背景 Disruptor是LMAX開發的一個高性能隊列,研發的初衷是解決記憶體隊列的延遲問題(在性能測試中發現竟然與I/O操作處於同樣的數量級)。基於Disruptor開發的系統單線程能支撐每秒600萬訂單,2010年在QCon演講後,獲得了業界關註。2011年,企業應用軟體專家Martin Fowl ...


背景

Disruptor是LMAX開發的一個高性能隊列,研發的初衷是解決記憶體隊列的延遲問題(在性能測試中發現竟然與I/O操作處於同樣的數量級)。基於Disruptor開發的系統單線程能支撐每秒600萬訂單,2010年在QCon演講後,獲得了業界關註。2011年,企業應用軟體專家Martin Fowler專門撰寫長文介紹。同年它還獲得了Oracle官方的Duke大獎。

目前,包括Apache Storm、Camel、Log4j2在內的很多知名項目都應用了Disruptor以獲取高性能。

我們先知道disruptor是乾什麼的,然後筆者帶你們源碼搞一波,再來看看在log4j2中的運用。

一、Disruptor是什麼?

可以這樣總結,Disruptor是LMAX開源的、用於替代併發線程間數據交換的環形隊列的、基本無鎖的(只有部分等待策略存在)、高性能的線程間通訊框架。

Disruptor唯一可能遇到Java鎖的時候,就是在消費者等待可用事件進行消費時。而Disruptor為這個等待過程,編寫了包括使用鎖和不使用鎖的多種策略,可根據不同場景和需求進行選擇。

開源https://github.com/LMAX-Exchange/disruptor

二、Disruptor為什麼這麼快

1、環形隊列RingBuffer

一個環形隊列,意味著首尾相連,對列可以迴圈使用,使用數組來保存。環形隊列在JVM生命周期中通常是永生的,GC的壓力更小

 

我們來解釋一下這個圖:當前有一個consumer,停留在位置12,這時producer假設在位置3,這時producer的下一步是如何處理的呢?producer會嘗試讀取4,發現下一個可以獲取,所以可以安全獲取,並且通知一個阻塞的consumer起來活動。如此一直到下一圈11都是安全的(這裡我們假設生產者比較快),當producer嘗試訪問12時發現不能繼續,於是自旋等待;當consumer消費時,會調用barrier的waitFor方法,waitFor看到前面最近的安全節點已經到了下一圈的11,於是consumer可以無鎖的去消費當前12到下一圈11所有數據,可以想象,這種方式比起synchronized要快上很多倍。

2、棄用鎖機制使用CAS

在高度競爭的情況下,鎖的性能將超過原子變數的性能,但是更真實的競爭情況下,原子變數的性能將超過鎖的性能。同時原子變數不會有死鎖等活躍性問題。能不用鎖,就不使用鎖,如果使用,也要將鎖的粒度最小化。

唯一使用鎖的就是消費者的等待策略實現類中,下圖。補充一句,生產者的等到策略就是LockSupport.parkNanos(1),再自旋判斷。

 

名稱措施適用場景
BlockingWaitStrategy 加鎖 CPU資源緊缺,吞吐量和延遲並不重要的場景
BusySpinWaitStrategy 自旋 通過不斷重試,減少切換線程導致的系統調用,而降低延遲。推薦線上程綁定到固定的CPU的場景下使用
PhasedBackoffWaitStrategy 自旋 + yield + 自定義策略 CPU資源緊缺,吞吐量和延遲並不重要的場景
SleepingWaitStrategy 自旋 + yield + sleep 性能和CPU資源之間有很好的折中。延遲不均勻
TimeoutBlockingWaitStrategy 加鎖,有超時限制 CPU資源緊缺,吞吐量和延遲並不重要的場景
YieldingWaitStrategy 自旋 + yield + 自旋 性能和CPU資源之間有很好的折中。延遲比較均勻

3、解決偽共用,採用緩存行填充

從上圖看到,線程1在CPU核心1上讀寫變數X,同時線程2在CPU核心2上讀寫變數Y,不幸的是變數X和變數Y在同一個緩存行上,每一個線程為了對緩存行進行讀寫,都要競爭並獲得緩存行的讀寫許可權,如果線程2在CPU核心2上獲得了對緩存行進行讀寫的許可權,那麼線程1必須刷新它的緩存後才能在核心1上獲得讀寫許可權,這導致這個緩存行在不同的線程間多次通過L3緩存來交換最新的拷貝數據,這極大的影響了多核心CPU的性能。 下麵代碼解決偽共用問題的,就是實例變數前後各加7個long形變數,用空間換時間。
abstract class SingleProducerSequencerPad extends AbstractSequencer
{
    protected long p1, p2, p3, p4, p5, p6, p7;

    SingleProducerSequencerPad(int bufferSize, WaitStrategy waitStrategy)
    {
        super(bufferSize, waitStrategy);
    }
}

public final class SingleProducerSequencer extends SingleProducerSequencerFields
{
    protected long p1, p2, p3, p4, p5, p6, p7;
    //..省略
}

Java中通過填充緩存行,來解決偽共用問題的思路,現在可能已經是老生常談,連Java8中都新增了sun.misc.Contended註解來避免偽共用問題。但在Disruptor剛出道那會兒,用緩存行來優化Java數據結構,這恐怕還很新潮。

4、還有一些細節性的

1)通過sequence & (bufferSize - 1)定位元素的index比普通的求餘取模(%)要快得多。sequence >>> indexShift 快速計算出sequence/bufferSize的商flag(其實相當於當前sequence在環形跑道上跑了幾圈,在數據生產時要設置好flag。

2)合理使用Unsafe,CPU級別指令。實現更加高效地記憶體管理和原子訪問。

至於一些更細節的,下麵源碼搞起來,還是很簡單的。

源碼分析:

正在搞。。。

參考:

https://tech.meituan.com/disruptor.html

https://www.jianshu.com/p/c3c108c3dcfd

 


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

-Advertisement-
Play Games
更多相關文章
  • 全局光照這個名詞在電腦圖形學里已經不算一個新名詞了,現在一提到擬真度,很多人基本上都會去想到全局光照,這個名詞上世紀七八十年代就有了,好像是由一個叫Jim Kajiya的大神在他那篇已經被引用了不知道多少次的論文里《The Rendering Equation》里提出來的,現在很多全局光照演算法基本 ...
  • Django中,與資料庫相關的模塊是model模塊,它提供了一種簡單易操作的API方式與資料庫交互,它是通過ORM映射的方式來操作資料庫,一個類對應資料庫一張表,一個類屬性,對應該表的一個欄位,一個實例化的類對象就是一個表中的一行數據信息。在開發的階段,工程師只需要python語言本身進行代碼設計, ...
  • re.match函數 re.match 嘗試從字元串的起始位置匹配一個模式,如果不是起始位置匹配成功的話,match()就返回none。 函數語法 參數說明: pattern 要匹配的正則表達式string 要匹配的字元串flags 標誌位,用於控制正則表達式的匹配方式,例如,是否要區分大小寫等等, ...
  • Python基礎知識(6):基本數據類型之列表 在Python中,最基本的數據結構是序列。序列中的每個元素被分配一個序號——即元素的位置,也稱為索引。第一個索引從0開始,如果要從右邊開始,序列中的最後一個元素標記為-1,倒數第二個標記為-2,以此類推。Python包含6種內建序列:字元串、列表、元組 ...
  • set.pop()理論上是隨機移除集合中的某一元素,但我在解釋器里試了4次,都是移除了集合中的第一個元素(PS: Python 3.6.2) ...
  • 一 .概述 先講緩存實現,主要是mybatis一級緩存,二級緩存及緩存使用後續補充 Mybatis緩存的實現是基於Map的,從緩存裡面讀寫數據是緩存模塊的核心基礎功能;除核心功能之外,有很多額外的附加功能,如:防止緩存擊穿,添加緩存清空策略(fifo、lru)、序列化功能、日誌能力、定時清空能力等; ...
  • 這幾天想做一個登陸界面,用Jframe做,連接資料庫時發現JPasswordField的getText()過時了,沒法使用。查了資料發現改成了: try{ String sql="SELECT * FROM username WHERE name=?"; conn = DB.getConnectio ...
  • 前面通過main方法介紹了方法的定義形式,對於方法的輸入參數來說,還有幾個值得註意的地方,接下來分別對輸入參數的幾種用法進行闡述。一個方法可以有輸入參數,也可以沒有輸入參數,倘若無需輸入參數,則方法定義的圓括弧內部直接留空。以列印當前時間為例,下麵的showTime方法沒有輸入參數也能正常實現: 在 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...