java 如何知道对象是否被修改过_Java 并发编程:AQS 的原子性如何保证
當我們研究AQS框架時(對于AQS不太熟知可以先閱讀《什么是JDK內置并發框架AQS》,會發現AbstractQueuedSynchronizer這個類很多地方都使用了CAS操作。在并發實現中CAS操作必須具備原子性,而且是硬件級別的原子性。我們知道Java被隔離在硬件之上,硬件級別的操作明顯力不從心。這時為了能夠執行操作系統層面的操作,就必須要通過用C++編寫的native本地方法來擴展實現。一般可以通過JNI方式實現Java代碼調用C++代碼
Unsafe調用
JDK提供了一個類來滿足CAS的硬件級別原子性要求,即sun.misc.Unsafe類,從名字上大概知道它用于執行低級別、不安全的操作,AQS就是使用此類來完成硬件級別的原子操作。也就是說通過該類就能實現對處理器的原子操作,Unsafe通過JNI調用本地C++代碼,C++代碼調用了硬件指令集,這些硬件指令集都屬于CPU。
Unsafe的魔法
Unsafe是一個很強大的類,它可以分配內存、釋放內存、可以定位對象某字段的位置、可以修改對象的字段值、可以使線程掛起、使線程恢復、可進行硬件級別原子的CAS操作等等。
Unsafe的用途
一般情況下,我們都沒有特殊的需求去使用它。而且它的使用必須限定在受信任代碼,這些代碼一般由JVM指定。例如直接Unsafe unsafe = Unsafe.getUnsafe();獲取一個Unsafe實例是不會成功的,因為這個類的安全性很重要,設計者對其進行了安全檢查。如下代碼所示,它會檢測調用它的類是否由啟動類加載器Bootstrap ClassLoader(它的類加載器為null)加載,由此來保證此類只能由JVM指定的類使用。
Unsafe實現CAS
因為存在安全性問題,所以如果我們要用Unsafe類則需要另辟蹊徑。可行的方法就是通過反射來繞過上述的安全檢查,我們可以通過以下的getUnsafeInstance方法來獲取Unsafe實例。這段代碼演示了如何獲取Java對象的相對地址偏移量以及使用Unsafe來完成CAS操作,最終輸出的是flag字段的內存偏移量及CAS操作后的值。最終的輸出為“unsafeTest對象的flag字段的地址偏移量為:12”和“CAS操作后的flag值為:101”。另外如果使用開發工具如Eclipse,可能會編譯通不過,只要把編譯錯誤提示關掉即可。
總結
這里主要講解了Unsafe類如何讓Java層能實現硬件級別的原子操作,同時也了解了Unsafe類擁有很多法魔技能。通常我們使用Java時不需要在內存中處理Java對象及內存地址位置,但有的時候我們被迫必須要操作Java對象相關的地址,于是我們只能使用Unsafe類。使用該類則意味著破壞了Java平臺隔離的效果了,我們都知道一旦用了本地方法則可能會引來跨平臺問題。
Java 并發編程
- Java并發編程:AQS的原子性如何保證
- Java并發編程:如何防止在線程阻塞與喚醒時死鎖
- Java并發編程:多線程如何實現阻塞與喚醒
- Java并發編程:任務執行器Executor接口
- Java并發編程:并發中死鎖的形成條件及處理
- Java并發編程:Java 序列化的工作機制
- Java并發編程:進程、線程、并行與并發
總結
以上是生活随笔為你收集整理的java 如何知道对象是否被修改过_Java 并发编程:AQS 的原子性如何保证的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: Linux如何修改主机名
- 下一篇: java 多线程同步问题_Java多线程