java unsafe获取指针_【实战Java高并发程序设计 1】Java中的指针:Unsafe类
是《實(shí)戰(zhàn)Java高并發(fā)程序設(shè)計(jì)》第4章的幾點(diǎn)。
如果你對技術(shù)有著不折不撓的追求,應(yīng)該還會(huì)特別在意incrementAndGet() 方法中compareAndSet()的實(shí)現(xiàn)?,F(xiàn)在,就讓我們更進(jìn)一步看一下它把!
public?final?boolean?compareAndSet(int?expect,?int?update){
returnunsafe.compareAndSwapInt(this,?valueOffset,?expect,?update);
}
在這里,我們看到一個(gè)特殊的變量unsafe。它是sun.misc.Unsafe類型。從名字看,這個(gè)類應(yīng)該是封裝了一些不安全的操作。那什么操作是不安全的呢?學(xué)習(xí)過C或者C++的話,大家應(yīng)該知道,指針是不安全的。這也是在Java中把指針去除的重要原因。如果指針指錯(cuò)了位置,或者計(jì)算指針偏移量時(shí)出錯(cuò),結(jié)果可能是災(zāi)難性的,你很有可能會(huì)覆蓋別人的內(nèi)存,導(dǎo)致系統(tǒng)奔潰。
而這里的Unsafe就是封裝了一些類似指針的操作。compareAndSwapInt()方法是一個(gè)navtive方法。它的幾個(gè)參數(shù)含義如下:
public?final?native?boolean?compareAndSwapInt(Object?o,long?offset,int?expected,int?x);
第一個(gè)參數(shù)o為給定的對象,offset為對象內(nèi)的偏移量(其實(shí)就是一個(gè)字段到對象頭部的偏移量,通過這個(gè)偏移量可以快速定位字段),expected表示期望值,x表示要設(shè)置的值。如果指定的字段的值等于expected,那么就會(huì)把它設(shè)置為x。
不難看出,compareAndSwapInt()方法的內(nèi)部,必然是使用CAS原子指令來完成的。此外,Unsafe類還提供了一些方法,主要有以下幾個(gè)(以Int操作為例,其他數(shù)據(jù)類型是類似的):
//獲得給定對象偏移量上的int值
public?native?int?getInt(Object?o,?long?offset);
//設(shè)置給定對象偏移量上的int值
public?native?void?putInt(Object?o,?long?offset,?int?x);
//獲得字段在對象中的偏移量
public?native?long?objectFieldOffset(Field?f);
//設(shè)置給定對象的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,?intx);
如果大家還記得“3.3.4 深度剖析ConcurrentLinkedQueue”一節(jié)中的描述的ConcurrentLinkedQueue實(shí)現(xiàn),應(yīng)該對ConcurrentLinkedQueue中的Node還有些印象。Node一些CAS操作也都是使用Unsafe類來實(shí)現(xiàn)的。大家可以回顧一下,以加深對Unsafe類的印象。
這里就可以看到,雖然Java拋棄了指針。但是在關(guān)鍵時(shí)刻,類似指針的技術(shù)還是必不可少的。這里底層的Unsafe實(shí)現(xiàn)就是最好的例子。但是很不幸,JDK的開發(fā)人員并不希望大家使用這個(gè)類。獲得Unsafe實(shí)例的方法是調(diào)動(dòng)其工廠方法getUnsafe()。但是,它的實(shí)現(xiàn)卻是這樣:
public?static?Unsafe?getUnsafe()?{
Class?cc?=Reflection.getCallerClass();
if(cc.getClassLoader()?!=?null)
throw?newSecurityException("Unsafe");
return?theUnsafe;
}
注意加粗部分的代碼,它會(huì)檢查調(diào)用getUnsafe()函數(shù)的類,如果這個(gè)類的ClassLoader不為null,就直接拋出異常,拒絕工作。因此,這也使得我們自己的應(yīng)用程序無法直接使用Unsafe類。它是一個(gè)JDK內(nèi)部使用的專屬類。
注意:根據(jù)Java 類加載器的工作原理,應(yīng)用程序的類由AppLoader加載。而系統(tǒng)核心類,如rt.jar中的類由Bootstrap類加載器加載。Bootstrap加載器沒有Java對象的對象,因此試圖獲得這個(gè)類加載器會(huì)返回null。所以,當(dāng)一個(gè)類的類加載器為null時(shí),說明它是由Bootstrap加載的,而這個(gè)類也極有可能是rt.jar中的類。
這本書:
總結(jié)
以上是生活随笔為你收集整理的java unsafe获取指针_【实战Java高并发程序设计 1】Java中的指针:Unsafe类的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: “林塘得芳景”下一句是什么
- 下一篇: java录排名怎么写_面试官:Java排