CAS無鎖策略

来源:https://www.cnblogs.com/sunshine-ground-poems/archive/2019/01/22/10306650.html
-Advertisement-
Play Games

在併發編程中,對於共用資源的使用需要確保絕對的安全性。除了利用鎖機制之外,還有一種無鎖的概念。所謂無鎖,就是假定在併發情況下,對於共用資源的訪問沒有衝突,線程可以一直不停的運行,無需阻塞,如果產生衝突,則使用CAS演算法確保全全性。Java在很多併發代碼中都使用了這種演算法。 CAS演算法的核心參數如下: ...


  併發編程時,對於共用資源的使用需要確保絕對的安全性。除了利用鎖機制之外,還有一種無鎖的概念。所謂無鎖,就是假定在併發情況下,對於共用資源的訪問沒有衝突,線程可以一直不停的運行,無需阻塞,如果產生衝突,則使用CAS演算法確保全全性。Java在很多併發代碼中都使用了這種演算法。

  CAS演算法的核心參數如下:

compareAndSet(V,E,A)

  V代碼需要進行更新的變數;E代表預期值;A代表所要更新的值。

  CAS的核心思想就是:當要對一個變數進行更新時,先取出該變數此時在記憶體中的實際值,與預期值進行比較。如果相等,則代表沒有其他線程對其進行過修改,直接更新。如果不同,表示有其他線程修改過,此時有兩種策略。一種是讓CAS自旋,直到更新成功;另外表示當前線程更新失敗,繼續後續邏輯。大概示意圖如下:

  對於CAS自旋,有可能會出現長時間更新失敗,浪費CPU性能的情況。JDK1.6以後,加入了適應性自旋的概念。即如果某個鎖自旋時很少獲得成功,那麼會減少自旋次數。自旋次數的預設值是10次,可以使用參數-XX:PreBlockSpin來更改。

  再說一下為什麼CAS在更新變數值的時候不會受到其他線程的影響呢?會不會我在判斷更新值與預期值相等,進行更新的過程中,變數被其他線程更新了呢?其實不會,因為CAS具有排它性,一次CAS是一個原子操作,是CPU源語級別。確保隔離性。

  Java對於CAS是使用Unsafe類進行支持的。顧名思義,不安全,可以讓程式員像C語言那樣直接操縱記憶體指針。java.util.concurrent.atomic包下的類,都是使用CAS實現的。

 

ABA缺陷

  不知道大家通過上面對CAS的介紹說明,看到了CAS演算法的缺陷沒有!

  CAS本身會有ABA問題。舉個例子,變數m = 10,我要把m更新為30。在我判斷之前,有其他線程先把m更新到50,在從50更新到10。那麼我進行 10 == 10的判斷時,這時候我怎麼確定m的值有沒有被改過呢?這就是ABA問題了。

  要解決這個問題,我們就需要在進行 10 == 10判斷的時候,還需要有其他參照。Java中有一個類AtomicStampedReference,這個類除了維護變數值之外,還會維護一個時間戳。用於確定數值版本。

private static class Pair<T> {
        final T reference;
        final int stamp;
        private Pair(T reference, int stamp) {
            this.reference = reference;
            this.stamp = stamp;
        }
        static <T> Pair<T> of(T reference, int stamp) {
            return new Pair<T>(reference, stamp);
        }
    }

    private volatile Pair<V> pair;

  用volatile關鍵字修飾的內部類。

  

  看一下核心的compareAndSet()方法:

public boolean compareAndSet(V   expectedReference,
                                 V   newReference,
                                 int expectedStamp,
                                 int newStamp) {
        Pair<V> current = pair;
        return
            expectedReference == current.reference &&
            expectedStamp == current.stamp &&
            ((newReference == current.reference &&
              newStamp == current.stamp) ||
             casPair(current, Pair.of(newReference, newStamp)));
    }

  可以看到除了比較數值之外,還會比較時間戳。

 


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

-Advertisement-
Play Games
更多相關文章
  • Java反射,註解,以及動態代理 基礎 最近在準備實習面試,被學長問到了Java反射,註解和動態代理的內容,發現有點自己有點懵,這幾天查了很多資料,就來說下自己的理解吧【如有錯誤,望指正】 Java反射 首先,我們得弄清一個,什麼是反射(Reflection)。簡單的來說,反射就是讓我們在程式運行的 ...
  • 實現跳轉的方法: 1.Php中header的函數 2js中location函數 3.Html中的meta函數 引入message.html <meta http-equiv="Refresh" content ="<?php echo $wait;?> ;url=<?php echo $url;?> ...
  • 1.簡介 HashMap是基於哈希表的Map介面的實現,用來存放鍵值對(Entry<Key,Value>),並提供可選的映射操作。使用put(Key,Value)存儲對象到HashMap中,使用get(Key)從hashMap中獲取對象。 2.底層結構 HashMap的底層是由數組加鏈表實現的,因為 ...
  • Java 12將在兩個月後(2019/3/19)發佈,現已進入RDP1階段,確定加入8個JEP。其中對Java語法的改進是JEP 325: switch表達式。於是我迫不及待,提前感受一下更先進的語言特性。 ...
  • 2019-01-22 22:50:05 centos7預設安裝的是python2.7,然而python2基本上要淘汰了,所以有必要安裝最新的python3 python,g++這些工具一般安裝在/usr/bin目錄里 通過指令ll python*可以看到python指向的是python2.7 我們要 ...
  • Python3新特性 類型註解 以及 點點點 ... + Python3 的 新特性 + Python 是一種動態語言,變數以及函數的參數是 不區分類型 的 + 在 函數中使用類型註解 相當於 給 形參的 類型 設置了一個備註 + 使用 PyCharm 編寫python代碼時 函數調用會有預設參數的 ...
  • 【問題描述】 輸出1到n之間所有不重覆的排列,即1到n的全排,要求所產生的任一數列不含有重覆的數字. 【代碼展示】 #include<iostream>using namespace std;int a[100],b[100];void quanpai(int index,int n){ //遞歸邊 ...
  • 1、二維數組的定義:當數組中每個元素帶有兩個下標時,稱這樣的數組為二維數組。在邏輯上可以把二維數組看成是一個具有行和列的表格或一個矩陣。 一般形式:類型說明符 數組名[常量表達式1][常量表達式2]; 例:定義a為3*4(3行4列)的數組,b為5*10(5行10列)的數組。 在記憶體中的表達: 例如: ...
一周排行
    -Advertisement-
    Play Games
  • .Net8.0 Blazor Hybird 桌面端 (WPF/Winform) 實測可以完整運行在 win7sp1/win10/win11. 如果用其他工具打包,還可以運行在mac/linux下, 傳送門BlazorHybrid 發佈為無依賴包方式 安裝 WebView2Runtime 1.57 M ...
  • 目錄前言PostgreSql安裝測試額外Nuget安裝Person.cs模擬運行Navicate連postgresql解決方案Garnet為什麼要選擇Garnet而不是RedisRedis不再開源Windows版的Redis是由微軟維護的Windows Redis版本老舊,後續可能不再更新Garne ...
  • C#TMS系統代碼-聯表報表學習 領導被裁了之後很快就有人上任了,幾乎是無縫銜接,很難讓我不想到這早就決定好了。我的職責沒有任何變化。感受下來這個系統封裝程度很高,我只要會調用方法就行。這個系統交付之後不會有太多問題,更多應該是做小需求,有大的開發任務應該也是第二期的事,嗯?怎麼感覺我變成運維了?而 ...
  • 我在隨筆《EAV模型(實體-屬性-值)的設計和低代碼的處理方案(1)》中介紹了一些基本的EAV模型設計知識和基於Winform場景下低代碼(或者說無代碼)的一些實現思路,在本篇隨筆中,我們來分析一下這種針對通用業務,且只需定義就能構建業務模塊存儲和界面的解決方案,其中的數據查詢處理的操作。 ...
  • 對某個遠程伺服器啟用和設置NTP服務(Windows系統) 打開註冊表 HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\Services\W32Time\TimeProviders\NtpServer 將 Enabled 的值設置為 1,這將啟用NTP伺服器功 ...
  • title: Django信號與擴展:深入理解與實踐 date: 2024/5/15 22:40:52 updated: 2024/5/15 22:40:52 categories: 後端開發 tags: Django 信號 松耦合 觀察者 擴展 安全 性能 第一部分:Django信號基礎 Djan ...
  • 使用xadmin2遇到的問題&解決 環境配置: 使用的模塊版本: 關聯的包 Django 3.2.15 mysqlclient 2.2.4 xadmin 2.0.1 django-crispy-forms >= 1.6.0 django-import-export >= 0.5.1 django-r ...
  • 今天我打算整點兒不一樣的內容,通過之前學習的TransformerMap和LazyMap鏈,想搞點不一樣的,所以我關註了另外一條鏈DefaultedMap鏈,主要調用鏈為: 調用鏈詳細描述: ObjectInputStream.readObject() DefaultedMap.readObject ...
  • 後端應用級開發者該如何擁抱 AI GC?就是在這樣的一個大的浪潮下,我們的傳統的應用級開發者。我們該如何選擇職業或者是如何去快速轉型,跟上這樣的一個行業的一個浪潮? 0 AI金字塔模型 越往上它的整個難度就是職業機會也好,或者說是整個的這個運作也好,它的難度會越大,然後越往下機會就會越多,所以這是一 ...
  • @Autowired是Spring框架提供的註解,@Resource是Java EE 5規範提供的註解。 @Autowired預設按照類型自動裝配,而@Resource預設按照名稱自動裝配。 @Autowired支持@Qualifier註解來指定裝配哪一個具有相同類型的bean,而@Resourc... ...