從上一節裡,我們學習到JNI的方法表示可知,GetStaticMethodID方法最後一個參數“([Ljava/lang/String;)V”。這個參數是一個字串,但內容排列比較奇怪,其實它是一種對函數傳回值和參數的編碼。這種編碼叫做JNI欄位描述符(JavaNative
Interface FieldDescriptors)。這裡只是方法描述符,但也有類描述符。其實這串參數“([Ljava/lang/String;)V”具體是表示什麼意思呢?怎麼樣來理解它呢?那麼就需要學習這一節了,其實這串字串就是描述像下面的函數:
publicstatic void main(String argv[]);
在JNI裡,一個類的表示,不再是使用句點來表示,而使用斜杠來表示的,比如類
java.lang.String 表示為java/lang/String的樣子。
一個數組int[],就需要表示為這樣"[I"。如果多個數組double[][][]就需要表示為這樣
"[[[D"。也就是說每一個方括弧開始,就表示一個數組維數。多個方框後面,就是數組的類型。
java裡8種基本欄位描述符如下:
-
欄位描述符 |
Java語言類型 |
Z
|
boolean
|
B
|
byte
|
C
|
char
|
S
|
short
|
I
|
int
|
J
|
long
|
F
|
float
|
D
|
double
|
如果以一個L開頭的描述符,就是類描述符,它後緊跟著類的字串,然後分號“;”結束。
比如"Ljava/lang/String;"就是表示類型String;
"[I"就是表示int[];
"[Ljava/lang/Object;"就是表示Object[]。
JNI方法描述符,主要就是在括弧裡放置參數,在括弧後面放置傳回型別,如下:
(參數描述符)傳回型別
當一個函數不需要返回參數類型時,就使用”V”來表示。
比如"()Ljava/lang/String;"就是表示Stringf();
"(ILjava/lang/Class;)J"就是表示longf(int
i, Class c);
"([B)V"就是表示voidString(byte[]
bytes);
由上面可見,當多個參數在一起時,中間沒有任何的分隔,直接連接在一起。如果你已經寫好一個JNI調用類,還不知道怎麼樣手動編寫這個描述符時,可以使用下面的工具來生產:
javap -s -p 類檔案名稱
由於java語言是強型別語言,就算相同的方法名稱,不同的參數也是不同的方法。因此,當從C語言尋找java裡的方法時,一定要傳送參數所有類型,才可以尋找到相應的方法。又由於跨語言的調用,需要使用這種方法來描述參數類型,才達到上述的目標。
通過上面的學習,理解了JNI裡尋找方法的參數描述符,同時也理解JNI怎麼樣實現C與JAVA不同語言之間參數傳送,因此對理解Dalvik虛擬機器的實現是相當有用的。