摘要:不難看出,方法的內部,必然是使用原子指令來完成的。它是一個內部使用的專屬類。注意根據類加載器的工作原理,應用程序的類由加載。加載器沒有對象的對象,因此試圖獲得這個類加載器會返回。
如果你對技術有著不折不撓的追求,應該還會特別在意incrementAndGet() 方法中compareAndSet()的實現。現在,就讓我們更進一步看一下它把!
public final boolean compareAndSet(int expect, int update) {
return unsafe.compareAndSwapInt(this, valueOffset, expect, update);
}
在這里,我們看到一個特殊的變量unsafe。它是sun.misc.Unsafe類型。從名字看,這個類應該是封裝了一些不安全的操作。那什么操作是不安全的呢?學習過C或者C++的話,大家應該知道,指針是不安全的。這也是在Java中把指針去除的重要原因。如果指針指錯了位置,或者計算指針偏移量時出錯,結果可能是災難性的,你很有可能會覆蓋別人的內存,導致系統奔潰。
而這里的Unsafe就是封裝了一些類似指針的操作。compareAndSwapInt()方法是一個navtive方法。它的幾個參數含義如下:
public final native boolean compareAndSwapInt(Object o, long offset,int expected,int x);
第一個參數o為給定的對象,offset為對象內的偏移量(其實就是一個字段到對象頭部的偏移量,通過這個偏移量可以快速定位字段),expected表示期望值,x表示要設置的值。如果指定的字段的值等于expected,那么就會把它設置為x。
不難看出,compareAndSwapInt()方法的內部,必然是使用CAS原子指令來完成的。此外,Unsafe類還提供了一些方法,主要有以下幾個(以Int操作為例,其他數據類型是類似的):
//獲得給定對象偏移量上的int值 public native int getInt(Object o, long offset); //設置給定對象偏移量上的int值 public native void putInt(Object o, long offset, int x); //獲得字段在對象中的偏移量 public native long objectFieldOffset(Field f); //設置給定對象的int值,使用volatile語義 public native void putIntVolatile(Object o, long offset, int x); //獲得給定對象對象的int值,使用volatile語義 public native int getIntVolatile(Object o, long offset); //和putIntVolatile()一樣,但是它要求被操作字段就是volatile類型的 public native void putOrderedInt(Object o, long offset, int x); 如果大家還記得“3.3.4 深度剖析ConcurrentLinkedQueue”一節中的描述的ConcurrentLinkedQueue實現,應該對ConcurrentLinkedQueue中的Node還有些印象。Node一些CAS操作也都是使用Unsafe類來實現的。大家可以回顧一下,以加深對Unsafe類的印象。 這里就可以看到,雖然Java拋棄了指針。但是在關鍵時刻,類似指針的技術還是必不可少的。這里底層的Unsafe實現就是最好的例子。但是很不幸,JDK的開發人員并不希望大家使用這個類。獲得Unsafe實例的方法是調動其工廠方法getUnsafe()。但是,它的實現卻是這樣: public static Unsafe getUnsafe() { Class cc = Reflection.getCallerClass(); if (cc.getClassLoader() != null) throw new SecurityException(“Unsafe”); return theUnsafe; }
注意加粗部分的代碼,它會檢查調用getUnsafe()函數的類,如果這個類的ClassLoader不為null,就直接拋出異常,拒絕工作。因此,這也使得我們自己的應用程序無法直接使用Unsafe類。它是一個JDK內部使用的專屬類。
注意:根據Java 類加載器的工作原理,應用程序的類由App Loader加載。而系統核心類,如rt.jar中的類由Bootstrap類加載器加載。Bootstrap加載器沒有Java對象的對象,因此試圖獲得這個類加載器會返回null。所以,當一個類的類加載器為null時,說明它是由Bootstrap加載的,而這個類也極有可能是rt.jar中的類。
本書現已上市,QQ群交流:397196583
購買地址:http://item.jd.com/11800589.html
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://specialneedsforspecialkids.com/yun/65454.html
摘要:實戰高并發程序設計連載中的指針類和非常類似,不同之處就在于是對整數的封裝,而則對應普通的對象引用。這樣,當前線程就無法正確判斷這個對象究竟是否被修改過。摘自實戰高并發程序設計一書 【實戰Java高并發程序設計】連載1–Java中的指針:Unsafe類 AtomicReference和AtomicInteger非常類似,不同之處就在于AtomicInteger是對整數的封裝,而Atomi...
摘要:有時候,由于初期考慮不周,或者后期的需求變化,一些普通變量可能也會有線程安全的需求。它可以讓你在不改動或者極少改動原有代碼的基礎上,讓普通的變量也享受操作帶來的線程安全性,這樣你可以修改極少的代碼,來獲得線程安全的保證。 有時候,由于初期考慮不周,或者后期的需求變化,一些普通變量可能也會有線程安全的需求。如果改動不大,我們可以簡單地修改程序中每一個使用或者讀取這個變量的地方。但顯然,這...
摘要:在本例中,講述的無鎖來自于并發包我們將這個無鎖的稱為。在這里,我們使用二維數組來表示的內部存儲,如下變量存放所有的內部元素。為什么使用二維數組去實現一個一維的呢這是為了將來進行動態擴展時可以更加方便。 我們已經比較完整得介紹了有關無鎖的概念和使用方法。相對于有鎖的方法,使用無鎖的方式編程更加考驗一個程序員的耐心和智力。但是,無鎖帶來的好處也是顯而易見的,第一,在高并發的情況下,它比有鎖...
摘要:同時,也提供了一個基于的實現類,底層基于紅黑樹設計,是一種有序的。可以看成是并發版本的,但是和不同是,并不是基于紅黑樹實現的,其底層是一種類似跳表的結構。上述所有構造器都調用了方法方法將一些字段置初始化,然后將指針指向新創建的結點。 showImg(https://segmentfault.com/img/remote/1460000016201159); 本文首發于一世流云專欄:ht...
摘要:如問到是否使用某框架,實際是是問該框架的使用場景,有什么特點,和同類可框架對比一系列的問題。這兩個方向的區分點在于工作方向的側重點不同。 [TOC] 這是一份來自嗶哩嗶哩的Java面試Java面試 32個核心必考點完全解析(完) 課程預習 1.1 課程內容分為三個模塊 基礎模塊: 技術崗位與面試 計算機基礎 JVM原理 多線程 設計模式 數據結構與算法 應用模塊: 常用工具集 ...
閱讀 2781·2023-04-25 14:41
閱讀 2375·2021-11-23 09:51
閱讀 3674·2021-11-17 17:08
閱讀 1667·2021-10-18 13:31
閱讀 5528·2021-09-22 15:27
閱讀 910·2019-08-30 15:54
閱讀 2222·2019-08-30 13:16
閱讀 728·2019-08-29 17:04