JAVA調(diào)用C語言寫的SO文件
因?yàn)?a target="_self" style="word-break: break-all; color: #202859; line-height: normal !important;">工作需要寫一份SO文件,作為手機(jī)硬件IC讀卡和APK交互的橋梁,也就是中間件,看了網(wǎng)上有說到JNI接口技術(shù)實(shí)現(xiàn),這里轉(zhuǎn)載了一個(gè)實(shí)例
1 // 用JNI實(shí)現(xiàn) 2 // 實(shí)例: 3 4 // 創(chuàng)建HelloWorld.java 5 class HelloWorld 6 { 7 private native void print(); 8 public static void main(String[] args) 9 { 10 new HelloWorld().print(); 11 } 12 13 static 14 { 15 System.loadLibrary("HelloWorld"); 16 } 17 } 18 // 注意print方法的聲明,關(guān)鍵字native表明該方法是一個(gè)原生代碼實(shí)現(xiàn)的。另外注意static代碼段的System.loadLibrary調(diào)用,這段代碼表示在程序加載的時(shí)候,自動(dòng)加載libHelloWorld.so庫。 19 // 編譯HelloWorld.java 20 // 在命令行中運(yùn)行如下命令: 21 javac HelloWorld.java 22 // 在當(dāng)前文件夾編譯生成HelloWorld.class。 23 // 生成HelloWorld.h 24 // 在命令行中運(yùn)行如下命令: 25 javah -jni HelloWorld 26 // 在當(dāng)前文件夾中會(huì)生成HelloWorld.h。打開HelloWorld.h將會(huì)發(fā)現(xiàn)如下代碼: 27 /* DO NOT EDIT THIS FILE - it is machine generated */ 28 #include <jni.h> 29 /* Header for class HelloWorld */ 30 31 #ifndef _Included_HelloWorld 32 #define _Included_HelloWorld 33 #ifdef __cplusplus 34 extern "C" { 35 #endif 36 /* 37 * Class: HelloWorld 38 * Method: print 39 * Signature: ()V 40 */ 41 JNIEXPORT void JNICALL Java_HelloWorld_print 42 (JNIEnv *, jobject); 43 44 #ifdef __cplusplus 45 } 46 #endif 47 #endif 48 // 該文件中包含了一個(gè)函數(shù)Java_HelloWorld_print的聲明。這里面包含兩個(gè)參數(shù),非常重要,后面講實(shí)現(xiàn)的時(shí)候會(huì)講到。 49 // 實(shí)現(xiàn)HelloWorld.c 50 // 創(chuàng)建HelloWorld.c文件輸入如下的代碼: 51 #include <jni.h> 52 #include <stdio.h> 53 #include "HelloWorld.h" 54 55 JNIEXPORT void JNICALL 56 Java_HelloWorld_print(JNIEnv *env, jobject obj) 57 { 58 printf("Hello World!\n"); 59 } 60 // 注意必須要包含jni.h頭文件,該文件中定義了JNI用到的各種類型,宏定義等。 61 // 另外需要注意Java_HelloWorld_print的兩個(gè)參數(shù),本例比較簡單,不需要用到這兩個(gè)參數(shù)。但是這兩個(gè)參數(shù)在JNI中非常重要。 62 // env代表java虛擬機(jī)環(huán)境,Java傳過來的參數(shù)和c有很大的不同,需要調(diào)用JVM提供的接口來轉(zhuǎn)換成C類型的,就是通過調(diào)用env方法來完成轉(zhuǎn)換的。 63 // obj代表調(diào)用的對象,相當(dāng)于c++的this。當(dāng)c函數(shù)需要改變調(diào)用對象成員變量時(shí),可以通過操作這個(gè)對象來完成。 64 // 編譯生成libHelloWorld.so 65 // 在Linux下執(zhí)行如下命令來完成編譯工作: 66 cc -I/usr/lib/jvm/java-6-sun/include/linux/ 67 -I/usr/lib/jvm/java-6-sun/include/ 68 -fPIC -shared -o libHelloWorld.so HelloWorld.c 69 // 在當(dāng)前目錄生成libHelloWorld.so。注意一定需要包含Java的include目錄(請根據(jù)自己系統(tǒng)環(huán)境設(shè)定),因?yàn)镠elloworld.c中包含了jni.h。 70 // 另外一個(gè)值得注意的是在HelloWorld.java中我們LoadLibrary方法加載的是“HelloWorld”,可我們生成的Library卻是libHelloWorld。這是Linux的鏈接規(guī)定的,一個(gè)庫的必須要是:lib+庫名+.so。鏈接的時(shí)候只需要提供庫名就可以了。 71 // 運(yùn)行Java程序HelloWorld 72 // 大功告成最后一步,驗(yàn)證前面的成果的時(shí)刻到了: 73 java HelloWorld 74 // 如果你這步發(fā)生問題,如果這步你收到j(luò)ava.lang.UnsatisfiedLinkError異常,可以通過如下方式指明共享庫的路徑: 75 java -Djava.library.path='.' HelloWorld 76 // 當(dāng)然還有其他的方式可以指明路徑請參考《在Linux平臺下使用JNI》。 77 // 我們可以看到久違的“Hello world!”輸出了。 |
試著去完成,自己生成了一份com_test_GetMsg.h頭文件,并完成test.c,生成libtest.so文件,JAVA調(diào)用SO文件時(shí),屢次報(bào):
failed: Cannot load library: load_library(linker.cpp:761): not a valid ELF executable: /data/app-lib/com.example.iccommtest-libtest.so
也就是提供的SO無法load,是valid的。
注意,剛才引用的實(shí)例是JAVA調(diào)用SO,而我需要的是android調(diào)用SO,不然會(huì)頻繁上面錯(cuò)誤。
原因有兩點(diǎn):
1、JAVA和android的虛擬環(huán)境不一樣
2、Linux和android的系統(tǒng)庫文件不一樣
這樣導(dǎo)致了在Linux下通過JNI標(biāo)準(zhǔn)命名方式編譯的SO文件,在android是調(diào)用失敗的,原因是Linux和android的系統(tǒng)庫不一樣,而生產(chǎn)的SO跟生產(chǎn)環(huán)境庫文件有依賴關(guān)系,然后搭建了NDK和Cywin環(huán)境,然后生產(chǎn)的SO可以被android調(diào)用,
那么SO文件就必須完全遵循JNI命名規(guī)則,方法名是這樣:
/* * Class: com_samples_jni_test * Method: GetMsg * Signature: ()V */ JNIEXPORT jstring JNICALL Java_com_samples_jni_test_GetMsg (JNIEnv *, jobject); |
通過NDK和Cywin生產(chǎn)libtest.so,android調(diào)用成功!
posted on 2014-11-17 10:37 順其自然EVO 閱讀(1485) 評論(0) 編輯 收藏 所屬分類: 測試學(xué)習(xí)專欄