溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Java中unsafe操作的示例分析

發(fā)布時間:2021-08-23 13:52:26 來源:億速云 閱讀:135 作者:小新 欄目:編程語言

這篇文章將為大家詳細講解有關(guān)Java中unsafe操作的示例分析,小編覺得挺實用的,因此分享給大家做個參考,希望大家閱讀完這篇文章后可以有所收獲。

Unsafe是Java無鎖操作的基石,在無鎖并發(fā)類中都少不了它們的身影,比如ConcurrentHashMap, ConcurrentLinkedQueue, 都是由Unsafe類來實現(xiàn)的。相對于與Java中的鎖,它基本無開銷,會原地等待。

1 compareAndSwap

/**
* 比較obj的offset處內(nèi)存位置中的值和期望的值,如果相同則更新。此更新是不可中斷的。
* 
* @param obj 需要更新的對象
* @param offset obj中整型field的偏移量
* @param expect 希望field中存在的值
* @param update 如果期望值expect與field的當前值相同,設(shè)置filed的值為這個新值
* @return 如果field的值被更改返回true
*/
public native boolean compareAndSwapInt(Object obj, long offset, int expect, int update);

這個就是著名的CAS操作了,分為三步來做

  1. 獲取obj對象中為offset的偏移值,這里假設(shè)為realVal

  2. 比較realVal和expect

  3. 如果相同,將該值更新為update,否則不更新

CAS家族還包括有,compareAndSwapObject(), compareAndSwapLong(), compareAndSwapInt()等等

用AtomicInteger中一個經(jīng)典的例子來說明:

public final int getAndAdd(int delta) {  
  return unsafe.getAndAddInt(this, valueOffset, delta);
}

//unsafe.getAndAddInt
public final int getAndAddInt(Object var1, long var2, int var4) {
  int var5;
  do {
  /**獲取原始值*/
    var5 = this.getIntVolatile(var1, var2);
  /**確認原始值沒有被其它線程修改時,再執(zhí)行更新var5+var4操作*/
  } while(!this.compareAndSwapInt(var1, var2, var5, var5 + var4));
  return var5;
}

2 putOrder

/***
  * Sets the value of the integer field at the specified offset in the
  * supplied object to the given value. This is an ordered or lazy
  * version of <code>putIntVolatile(Object,long,int)</code>, which
  * doesn't guarantee the immediate visibility of the change to other
  * threads. It is only really useful where the integer field is
  * <code>volatile</code>, and is thus expected to change unexpectedly.
  *
  * @param obj the object containing the field to modify.
  * @param offset the offset of the integer field within <code>obj</code>.
  * @param value the new value of the field.
  * @see #putIntVolatile(Object,long,int)
  */
 public native void putOrderedInt(Object obj, long offset, int value);

將obj對象的偏移量為offset的位置修改為value,因為Java中沒有內(nèi)存操作,而Unsafe的這個操作正好補充了內(nèi)存操作的不足。也可以用于數(shù)組操作,比如ConcurrentHashMap中就大量用到了該操作

 Segment<K,V> s0 =
    new Segment<K,V>(loadFactor, (int)(cap * loadFactor),
             (HashEntry<K,V>[])new HashEntry[cap]);
  Segment<K,V>[] ss = (Segment<K,V>[])new Segment[ssize];
  // 往數(shù)組下標為0的位置,寫入s0: ss[0]=s0
  UNSAFE.putOrderedObject(ss, SBASE, s0); // ordered write of segments[0]

需要注意的是obj需要設(shè)置為Volatile,否則對于其它線程會不可見

3 putXxxVolatile

/***
  * Sets the value of the integer field at the specified offset in the
  * supplied object to the given value, with volatile store semantics.
  *
  * @param obj the object containing the field to modify.
  * @param offset the offset of the integer field within <code>obj</code>.
  * @param value the new value of the field.
  */
 public native void putIntVolatile(Object obj, long offset, int value);

感覺和putOrderInt一樣,因為必須設(shè)置為Volatile,否則有什么用呢?

關(guān)于“Java中unsafe操作的示例分析”這篇文章就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,使各位可以學到更多知識,如果覺得文章不錯,請把它分享出去讓更多的人看到。

向AI問一下細節(jié)

免責聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI