解析Java的JNI編程中的對象引用與內存泄漏問題

JNI,Java Native Interface,是 native code 的編程接口。JNI 使 Java 代碼程序能夠與 native code 交互——在 Java 程序中調用 native code;在 native code 中嵌入 Java 虛擬機調用 Java 的代碼。
JNI 編程在軟件開發中運用普遍,其優點能夠歸結爲如下幾點:
利用 native code 的平臺相關性,在平臺相關的編程中彰顯優點。
對 native code 的代碼重用。
native code 底層操做,更加高效。
然而任何事物都具備兩面性,JNI 編程也一樣如此。程序員在使用 JNI 時應當認識到 JNI 編程中以下的幾點弊端,揚長避短,才能夠寫出更加完善、高性能的代碼:
從 Java 環境到 native code 的上下文切換耗時、低效。
JNI 編程,若是操做不當,可能引發 Java 虛擬機的崩潰。
JNI 編程,若是操做不當,可能引發內存泄漏。
JAVA 中的內存泄漏
JAVA 編程中的內存泄漏,從泄漏的內存位置角度能夠分爲兩種:JVM 中 Java Heap 的內存泄漏;JVM 內存中 native memory 的內存泄漏。java

局部和全局引用程序員

JNI將實例、數組類型暴露爲不透明的引用。native代碼從不會直接檢查一個不透明的引用指針的上下文,而是經過使用JNI函數來訪問由不透明的引用所指向的數據結構。由於只處理不透明的引用,這樣就不須要擔憂不一樣的java VM實現而致使的不一樣的內部對象的佈局。然而,仍是有必要了解一下JNI中不一樣種類的引用:
1)JNI 支持3中不透明的引用:局部引用、全局引用和弱全局引用。
2)局部和全局引用,有着各自不一樣的生命週期。局部引用可以被自動釋放,而全局引用和若全局引用在被程序員釋放以前,是一直有效的。
3)一個局部或者全局引用,使所說起的對象不能被垃圾回收。而弱全局引用,則容許說起的對象進行垃圾回收。
4)不是全部的引用均可以在全部上下文中使用的。例如:在一個建立返回引用native方法以後,使用一個局部引用,這是非法的。編程

那麼到底什麼是局部引用,什麼事全局引用,它們有什麼不一樣?數組

局部引用緩存

多數JNI函數都建立局部引用。例如JNI函數NewObject建立一個實例,而且返回一個指向該實例的局部引用。數據結構

局部引用只在建立它的native方法的動態上下文中有效,而且只在native方法的一次調用中有效。全部局部引用只在一個native方法的執行期間有效,在該方法返回時,它就被回收。多線程

在native方法中使用一個靜態變量來保存一個局部引用,以便在隨後的調用中使用該局部引用,這種方式是行不通的。例如如下例子,誤用了局部引用:
/* This code is illegal */ 
jstring  編程語言

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
MyNewString(JNIEnv *env, jchar *chars, jint len)
{
   static jclass stringClass = NULL;
   jmethodID cid;
   jcharArray elemArr;
   jstring result;
   if (stringClass == NULL) {
     stringClass = (*env)->FindClass(env, "java/lang/String" );
     if (stringClass == NULL) {
       return NULL; /* exception thrown */
     }
   }
   /* It is wrong to use the cached stringClass here,
     because it may be invalid. */
   cid = (*env)->GetMethodID(env, stringClass, "<init>" , "([C)V" );
   ...
   elemArr = (*env)->NewCharArray(env, len);
   ...
   result = (*env)->NewObject(env, stringClass, cid, elemArr);
   (*env)->DeleteLocalRef(env, elemArr);
   return result;
}

這種保存局部引用的方式是不正確的,由於FindClass()返回的是對java.lang.String的局部引用。這是由於,在native代碼從MyNewString返回退出時,VM 會釋放全部局部引用,包括存儲在stringClass變量中的指向類對象的引用。這樣當再次後繼調用MyNewString時,可能會訪問非法地址,致使內存被破壞,或者系統崩潰。函數

局部引用失效,有兩種方式:‘
1)系統會自動釋放局部變量。
2)程序員能夠顯示地管理局部引用的生命週期,例如調用DeleteLocalRef。工具

一個局部引用可能在被摧毀以前,被傳給多個native方法。例如,MyNewString中,返回一個由NewObject建立的字符串引用,它將由NewObject的調用者來決定是否釋放該引用。而在如下代碼中:

?
1
2
JNIEXPORT jstring JNICALL Java_C_f(JNIEnv *env, jobject this ) {
    char *c_str = ...<pre name= "code" class = "cpp" >   ... <pre name= "code" class = "cpp" > return MyNewString(c_str);<pre name= "code" class = "cpp" >}

在VM接收到來自Java_C_f的局部引用之後,將基礎字符串對象傳遞給ava_C_f的調用者,而後摧毀本來由MyNewString中調用的JNI函數NewObject所建立的局部引用。

局部對象只屬於建立它們的線程,只在該線程中有效。一個線程想要調用另外一個線程建立的局部引用是不被容許的。將一個局部引用保存到全局變量中,而後在其它線程中使用它,這是一種錯誤的編程。

全局引用

在一個native方法被屢次調用之間,可使用一個全局引用跨越它們。一個全局引用能夠跨越多個線程,而且在被程序員釋放以前,一致有效。和局部引用同樣,全局引用保證了所引用的對象不會被垃圾回收。

和局部引用不同(局部變量能夠由多數JNI函數建立),全局引用只能由一個JNI函數建立(NewGlobalRef)。下面是一個使用全局引用版本的MyNewString:
/* This code is OK */ 
jstring 

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
MyNewString(JNIEnv *env, jchar *chars, jint len)
{
   static jclass stringClass = NULL;
   ...
   if (stringClass == NULL) {
     jclass localRefCls =
       (*env)->FindClass(env, "java/lang/String" );
     if (localRefCls == NULL) {
       return NULL; /* exception thrown */
     }
     /* Create a global reference */
     stringClass = (*env)->NewGlobalRef(env, localRefCls);
     /* The local reference is no longer useful */
     (*env)->DeleteLocalRef(env, localRefCls);
     /* Is the global reference created successfully? */
     if (stringClass == NULL) {
       return NULL; /* out of memory exception thrown */
     }
   }
   ...
}


弱全局引用


弱全局引用是在java 2 SDK1.2纔出現的。它由NewGolableWeakRef函數建立,而且被DeleteGloablWeakRef函數摧毀。和全局引用同樣,它能夠跨native方法調用,也能夠跨越不一樣線程。可是和全局引用不一樣的是,它不阻止對基礎對象的垃圾回收。下面是弱全局引用版的MyNewString:

JNIEXPORT void JNICALL 

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
Java_mypkg_MyCls_f(JNIEnv *env, jobject self)
{
   static jclass myCls2 = NULL;
   if (myCls2 == NULL) {
     jclass myCls2Local =
       (*env)->FindClass(env, "mypkg/MyCls2" );
     if (myCls2Local == NULL) {
       return ; /* can't find class */
     }
     myCls2 = NewWeakGlobalRef(env, myCls2Local);
     if (myCls2 == NULL) {
       return; /* out of memory */
     }
   }
   ... /* use myCls2 */
}

弱全局引用在一個被native代碼緩存着的引用不想阻止基礎對象被垃圾回收時,很是有用。如以上例子,mypkg.MyCls.f須要緩存mypkg.MyCls2的引用。而經過將mypkg.MyCls2緩存到弱引用中,可以實現MyCls2類依舊能夠被卸載。


上面代碼中,咱們假設了MyCls類和MyCls2類的生命週期是相同的(例如,在同一個類中被加載、卸載)。因此沒有考慮MyCls2被卸載了,而後在類MyCls和native方法的實現Java_mypkg_MyCls_f還要被繼續使用時,再被從新加載起來的狀況。針對於這個MyCls2類可能被卸載再加載的狀況,在使用時,須要檢查該弱全局引用是否還有效。如何檢查,這將在下面提到。

比較引用

能夠用JNI函數IsSameObject來檢查給定的兩個局部引用、全局引用或者弱全局引用,是否指向同一個對象。
(*env)->IsSameObject(env, obj1, obj2) 
返回值爲:
JNI_TRUE,表示兩個對象一致,是同一個對象。
JNI_FALSE,表示兩個對象不一致,不是同一個對象。


在java VM中NULL是null的引用。
若是一個對象obj是局部引用或者全局引用,則能夠這樣來檢查它是否指向null對象:

?
1
(*env)->IsSameObject(env, obj, NULL)

或者:

?
1
NULL == obj


而對於弱全局引用,以上規則須要改變一下:
咱們能夠用這個函數來判斷一個非0弱全局引用wobj所指向的對象是否仍舊存活着(依舊有效)。

?
1
(*env)->IsSameObject(env, wobj, NULL)

返回值:
JNI_TRUE,表示對象已經被回收了。
JNI_FALSE,表示wobj指向的對象,依舊有效。

 釋放引用
除了引用的對象要佔用內存,每一個JNI引用自己也會消耗必定內存。做爲一個JNI程序員,應該對在一段給定的時間裏,程序會用到的引用的個數,作到心中有數。特別是,儘管程序所建立的局部引用最終會被VM會被自動地釋放,仍舊須要知道在程序在執行期間的任什麼時候刻,建立的局部引用的上限個數。建立過多的引用,即使他們是瞬間、短暫的,也會致使內存耗盡。

釋放局部引用
多數狀況下,在執行一個native方法時,你不須要擔憂局部引用的釋放,java VM會在native方法返回調用者的時候釋放。然而有時候須要JNI程序員顯示的釋放局部引用,來避免太高的內存使用。那麼何時須要顯示的釋放呢,且看一下情景:
1)在單個native方法調用中,建立了大量的局部引用。這可能會致使JNI局部引用表溢出。此時有必要及時地刪除那些再也不被使用的局部引用。例如如下代碼,在該循環中,每次都有可能建立一個巨大的字符串數組。在每一個迭代以後,native代碼須要顯示地釋放指向字符串元素的局部引用:

?
1
2
3
4
5
for (i = 0 ; i < len; i++) {
   jstring jstr = (*env)->GetObjectArrayElement(env, arr, i);
   ... /* process jstr */
   (*env)->DeleteLocalRef(env, jstr);
}

2)你可能要建立一個工具函數,它會被未知的上下文調用。例如以前提到到MyNewString這個例子,它在每次返回調用者欠,都及時地將局部引用釋放。


3)native方法,可能不會返回(例如,一個可能進入無限事件分發的循環中的方法)。此時在循環中釋放局部引用,是相當重要的,這樣才能不會無限期地累積,進而致使內存泄露。


4)native方法可能訪問一個巨大的對象,所以,建立了一個指向該對象的局部引用。native方法在返回調用者以前,除訪問對象以外,還執行了額外的計算。指向這個大對象的局部引用,將會包含該對象,以防被垃圾回收。這個現象會持續到native方法返回到調用者時,即使這個對象不會再被使用,也依舊會受保護。在如下例子中,因爲在lengthyComputation()前,顯示地調用了DeleteLocalRef,因此垃圾回收器有機會能夠釋放lref所指向的對象。

?
1
2
3
4
5
6
7
8
9
10
/* A native method implementation */
JNIEXPORT void JNICALL
Java_pkg_Cls_func(JNIEnv *env, jobject this)
{
   lref = ...       /* a large Java object */
   ...           /* last use of lref */
   (*env)->DeleteLocalRef(env, lref);
   lengthyComputation();  /* may take some time */
   return;         /* all local refs are freed */
}

這個情形的實質,就是容許程序在native方法執行期間,java的垃圾回收機制有機會回收native代碼不在訪問的對象。

管理局部引用
不知道java 7怎麼樣了,應該更強大吧,有時間,去看看,這裏且按照java2的特性來吧。
SDK1.2中提供了一組額外的函數來管理局部引用的生命週期。他們是EnsureLocalCapacity、NewLocalRef、PushLocalFram以及PopLocalFram。
JNI的規範要求VM能夠自動確保每一個native方法能夠建立至少16個局部引用。經驗顯示,若是native方法中未包含和java VM的對象進行復雜的互相操做,這個容量對大多數native方法而言,已經足夠了。若是,出現這還不夠的狀況,須要建立更多的局部引用,那麼native方法能夠調用EnsureLocalCapacity來保證這些局部引用有足夠的空間。

?
1
2
3
4
5
6
/* The number of local references to be created is equal to
   the length of the array. */
if ((*env)->EnsureLocalCapacity(env, len)) < 0) {   ... /* out of memory */
} for (i = 0; i < len; i++) {
   jstring jstr = (*env)->GetObjectArrayElement(env, arr, i);
   ... /* process jstr */   /* DeleteLocalRef is no longer necessary */ }
這樣作,所消耗的內存,天然就有可能比以前的版原本的多。


另外,PushLocalFram\PopLocalFram函數容許程序員建立嵌套做用域的局部引用。以下代碼:

?
1
2
3
4
5
6
7
8
9
10
#define N_REFS ... /* the maximum number of local references
  used in each iteration */
for (i = 0; i < len; i++) {
   if ((*env)->PushLocalFrame(env, N_REFS) < 0) {
     ... /* out of memory */
   }
   jstr = (*env)->GetObjectArrayElement(env, arr, i);
   ... /* process jstr */
   (*env)->PopLocalFrame(env, NULL);
}

PushLocalFram爲指定數目的局部引用,建立一個新的做用域,PopLocalFram摧毀最上層的做用域,而且釋放該域中的全部局部引用。


使用這兩個函數的好處是它們能夠管理局部引用的生命週期,而不需關係在執行過程當中可能被建立的每一個單獨局部引用。例子中,若是處理jstr的過程,建立了額外的局部引用,它們也會在PopLocalFram以後被當即釋放。


NewLocalRef函數,在你寫一個工具函數時,很是有用。這個會在下面章節——管理引用的規則,具體分析。

 

native代碼可能會建立超出16個局部引用的範圍,也可能將他們保存在PushLocalFram或者EnsureLocalCapacity調用,VM會爲局部引用分配所須要的內存。然而,這些內存是否足夠,是沒有保證的。若是內存分配失敗,虛擬機將會退出。

 釋放全局引用


在native代碼再也不須要訪問一個全局引用的時候,應該調用DeleteGlobalRef來釋放它。若是調用這個函數失敗,Java VM將不會回收對應的對象。


在native代碼不在須要訪問一個弱全局引用的時候,應該調用DeleteWeakGlobalRef來釋放它。若是調用這個函數失敗了,java VM 仍舊將會回收對應的底層對象,可是,不會回收這個弱引用自己所消耗掉的內存。


 管理引用的規則
管理引用的目的是爲了清除不須要的內存佔用和對象保留。

整體來講,只有兩種類型的native代碼:直接實現native方法的函數,在二進制上下文中被使用的工具函數。

在寫native方法的實現的時候,須要小心在循環中過分建立局部引用,以及在native方法中被建立的,卻不返回給調用者的局部引用。在native方法方法返回後還留有16個局部引用在使用中,將它們交給java VM來釋放,這是能夠接受的。可是native方法的調用,不該該引發全局引用和弱全局引用的累積。應爲這些引用不會在native方法返後被自動地釋放。


在寫工具函數的時候,必需要注意不能泄露任何局部引用或者超出該函數以外的執行。由於一個工具函數,可能在乎料以外的上下文中,被不停的重複調用。任何不須要的引用建立都有可能致使內存泄露。
1)當一個返回一個基礎類型的工具函數被調用,它必須應該沒有局部引用、若全局引用的累積。
2)當一個返回一個引用類型的工具函數被調用,它必須應該沒有局部、全局或若全局引用的累積,除了要被做爲返回值的引用。


一個工具函數以捕獲爲目的建立一些全局或者弱全局引用,這是可接受的,由於只有在最開始的時候,纔會建立這些引用。


若是一個工具函數返回一個引用,你應該使返回的引用的類型(例如局部引用、全局引用)做爲函數規範的一部分。它應該始終如一,而不是有時候返回一個局部引用,有時候卻返回一個全局引用。調用者須要知道工具函數返回的引用的類型,以便正確地管理本身的JNI引用。如下代碼重複地調用一個工具工具函數(GetInfoString)。咱們須要知道GetInfoString返回的引用的類型,以便釋放該引用:

?
1
2
3
4
5
6
7
while (JNI_TRUE) {
   jstring infoString = GetInfoString(info);
   ... /* process infoString */
   ??? /* we need to call DeleteLocalRef, DeleteGlobalRef,
  or DeleteWeakGlobalRef depending on the type of
  reference returned by GetInfoString. */
}

在java2 SDK1.2中,NewLocalRef函數能夠用來保證一個工具函數一直返回一個局部引用。爲了說明這個問題,咱們對MyNewString作一些改動,它緩存了一個被頻繁請求的字符串(「CommonString」)到全局引用:

jstring 

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
MyNewString(JNIEnv *env, jchar *chars, jint len)
{
   static jstring result;
   /* wstrncmp compares two Unicode strings */
   if (wstrncmp("CommonString", chars, len) == 0) {
     /* refers to the global ref caching "CommonString" */
     static jstring cachedString = NULL;
     if (cachedString == NULL) {
       /* create cachedString for the first time */
       jstring cachedStringLocal = ... ;
       /* cache the result in a global reference */
       cachedString =
         (*env)->NewGlobalRef(env, cachedStringLocal);
     }
     return (*env)->NewLocalRef(env, cachedString);
   }
   ... /* create the string as a local reference and store in
  result as a local reference */
   return result;
}

正常的流程返回的時候局部引用。就像以前解釋的那樣,咱們必須將緩存字符保存到一個全局引用中,這樣就能夠在多個線程中調用native方法時,都能訪問它。

?
1
return (*env)->NewLocalRef(env, cachedString);

這條語句,建立了一個局部引用,它指向了緩存在全局引用的指向的統一對象。做爲和調用者的約定的一部分,MyNewString老是返回一個局部引用。


PushLocalFram、PopLocalFram函數用來管理局部引用的生命週期特別得方便。只須要在native函數的入口調用PushLocalFram,在函數退出時調用PopLocalFram,局部變量就會被釋放。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
jobject f(JNIEnv *env, ...)
{
   jobject result;
   if ((*env)->PushLocalFrame(env, 10 ) < 0 ) {
     /* frame not pushed, no PopLocalFrame needed */
     return NULL;
   }
   ...
   result = ...;
   if (...) {
     /* remember to pop local frame before return */
     result = (*env)->PopLocalFrame(env, result);
     return result;
   }
   ...
   result = (*env)->PopLocalFrame(env, result);
   /* normal return */
   return result;
}

PopLocalFram函數調用失敗時,可能會致使未定義的行爲,例如VM崩潰。

內存泄漏問題
Java Heap 的內存泄漏
Java 對象存儲在 JVM 進程空間中的 Java Heap 中,Java Heap 能夠在 JVM 運行過程當中動態變化。若是 Java 對象愈來愈多,佔據 Java Heap 的空間也愈來愈大,JVM 會在運行時擴充 Java Heap 的容量。若是 Java Heap 容量擴充到上限,而且在 GC 後仍然沒有足夠空間分配新的 Java 對象,便會拋出 out of memory 異常,致使 JVM 進程崩潰。
Java Heap 中 out of memory 異常的出現有兩種緣由——①程序過於龐大,導致過多 Java 對象的同時存在;②程序編寫的錯誤致使 Java Heap 內存泄漏。
多種緣由可能致使 Java Heap 內存泄漏。JNI 編程錯誤也可能致使 Java Heap 的內存泄漏。
JVM 中 native memory 的內存泄漏
從操做系統角度看,JVM 在運行時和其它進程沒有本質區別。在系統級別上,它們具備一樣的調度機制,一樣的內存分配方式,一樣的內存格局。
JVM 進程空間中,Java Heap 之外的內存空間稱爲 JVM 的 native memory。進程的不少資源都是存儲在 JVM 的 native memory 中,例如載入的代碼映像,線程的堆棧,線程的管理控制塊,JVM 的靜態數據、全局數據等等。也包括 JNI 程序中 native code 分配到的資源。
在 JVM 運行中,多數進程資源從 native memory 中動態分配。當愈來愈多的資源在 native memory 中分配,佔據愈來愈多 native memory 空間而且達到 native memory 上限時,JVM 會拋出異常,使 JVM 進程異常退出。而此時 Java Heap 每每尚未達到上限。
多種緣由可能致使 JVM 的 native memory 內存泄漏。例如 JVM 在運行中過多的線程被建立,而且在同時運行。JVM 爲線程分配的資源就可能耗盡 native memory 的容量。
JNI 編程錯誤也可能致使 native memory 的內存泄漏。對這個話題的討論是本文的重點。

JNI 編程實現了 native code 和 Java 程序的交互,所以 JNI 代碼編程既遵循 native code 編程語言的編程規則,同時也遵照 JNI 編程的文檔規範。在內存管理方面,native code 編程語言自己的內存管理機制依然要遵循,同時也要考慮 JNI 編程的內存管理。
本章簡單歸納 JNI 編程中顯而易見的內存泄漏。從 native code 編程語言自身的內存管理,和 JNI 規範附加的內存管理兩方面進行闡述。
Native Code 自己的內存泄漏
JNI 編程首先是一門具體的編程語言,或者 C 語言,或者 C++,或者彙編,或者其它 native 的編程語言。每門編程語言環境都實現了自身的內存管理機制。所以,JNI 程序開發者要遵循 native 語言自己的內存管理機制,避免形成內存泄漏。以 C 語言爲例,當用 malloc() 在進程堆中動態分配內存時,JNI 程序在使用完後,應當調用 free() 將內存釋放。總之,全部在 native 語言編程中應當注意的內存泄漏規則,在 JNI 編程中依然適應。
Native 語言自己引入的內存泄漏會形成 native memory 的內存,嚴重狀況下會形成 native memory 的 out of memory。
Global Reference 引入的內存泄漏
JNI 編程還要同時遵循 JNI 的規範標準,JVM 附加了 JNI 編程特有的內存管理機制。
JNI 中的 Local Reference 只在 native method 執行時存在,當 native method 執行完後自動失效。這種自動失效,使得對 Local Reference 的使用相對簡單,native method 執行完後,它們所引用的 Java 對象的 reference count 會相應減 1。不會形成 Java Heap 中 Java 對象的內存泄漏。
而 Global Reference 對 Java 對象的引用一直有效,所以它們引用的 Java 對象會一直存在 Java Heap 中。程序員在使用 Global Reference 時,須要仔細維護對 Global Reference 的使用。若是必定要使用 Global Reference,務必確保在不用的時候刪除。就像在 C 語言中,調用 malloc() 動態分配一塊內存以後,調用 free() 釋放同樣。不然,Global Reference 引用的 Java 對象將永遠停留在 Java Heap 中,形成 Java Heap 的內存泄漏。

 

總結

 至此咱們把JNI規範中的三種引用都進行了一個簡單的介紹,在此我對以上內容作一個簡單總結:

一、局部引用是Native代碼中最經常使用的引用。大部分局部引用都是經過JNI API返回來建立,也能夠經過調用NewLocalRef來建立。另外強烈建議Native函數返回值爲局部引用。局部引用只在當前調用上下文中有效,因此局部引用不能用Native代碼中的靜態變量和全局變量來保存。另外時刻要記着Java虛擬機局部引用的個數是有限的,編程的時候強烈建議調用EnsureLocalCapacity,PushLocalFrame和PopLocalFrame來確保Native代碼可以得到足夠的局部引用數量。

二、全局變量必需要經過NewGlobalRef建立,經過DeleteGlobalRef刪除。主要用來緩存Field ID和Method ID。全局引用能夠在多線程之間共享其指向的對象。在C語言中以靜態變量和全局變量來保存。

三、全局引用和局部引用能夠阻止Java虛擬機回收其指向的對象。

四、弱全局引用必需要經過NewWeakGlobalRef建立,經過DeleteWeakGlobalRef銷燬。能夠在多線程之間共享其指向的對象。在C語言中經過靜態變量和全局變量來保持弱全局引用。弱全局引用指向的對象隨時均可能會被Java虛擬機回收,因此使用的時候須要時刻注意檢查其有效性。弱全局引用常常用來緩存jclass對象。

五、全局引用和弱全局引用能夠在多線程中共享其指向對象,可是在多線程編程中須要注意多線程同步。強烈建議在JNI_OnLoad初始化全局引用和弱全局引用,而後在多線程中進行讀全局引用和弱全局引用,這樣不須要對全局引用和弱全局引用同步(只有讀操做不會出現不一致狀況)。

 

轉自:http://www.jb51.net/article/75222.htm

相關文章
相關標籤/搜索