JAVA調用C語言寫的SO文件
因為工作需要寫一份SO文件,作為手機硬件IC讀卡和APK交互的橋梁,也就是中間件,看了網上有說到JNI接口技術實現,這里轉載了一個實例
1 // 用JNI實現 2 // 實例: 3 4 // 創建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 static14 {15 System.loadLibrary("HelloWorld");16 }17 }18 // 注意print方法的聲明,關鍵字native表明該方法是一個原生代碼實現的。另外注意static代碼段的System.loadLibrary調用,這段代碼表示在程序加載的時候,自動加載libHelloWorld.so庫。19 // 編譯HelloWorld.java20 // 在命令行中運行如下命令:21 javac HelloWorld.java22 // 在當前文件夾編譯生成HelloWorld.class。23 // 生成HelloWorld.h24 // 在命令行中運行如下命令:25 javah -jni HelloWorld26 // 在當前文件夾中會生成HelloWorld.h。打開HelloWorld.h將會發現如下代碼:27 /* DO NOT EDIT THIS FILE - it is machine generated */28 #include <jni.h>29 /* Header for class HelloWorld */30 31 #ifndef _Included_HelloWorld32 #define _Included_HelloWorld33 #ifdef __cplusplus34 extern "C" {35 #endif36 /*37 * Class: HelloWorld38 * Method: print39 * Signature: ()V40 */41 JNIEXPORT void JNICALL Java_HelloWorld_print42 (JNIEnv *, jobject);43 44 #ifdef __cplusplus45 }46 #endif47 #endif48 // 該文件中包含了一個函數Java_HelloWorld_print的聲明。這里面包含兩個參數,非常重要,后面講實現的時候會講到。49 // 實現HelloWorld.c50 // 創建HelloWorld.c文件輸入如下的代碼:51 #include <jni.h>52 #include <stdio.h>53 #include "HelloWorld.h"54 55 JNIEXPORT void JNICALL56 Java_HelloWorld_print(JNIEnv *env, jobject obj)57 {58 printf("Hello World!/n");59 }60 // 注意必須要包含jni.h頭文件,該文件中定義了JNI用到的各種類型,宏定義等。61 // 另外需要注意Java_HelloWorld_print的兩個參數,本例比較簡單,不需要用到這兩個參數。但是這兩個參數在JNI中非常重要。62 // env代表java虛擬機環境,Java傳過來的參數和c有很大的不同,需要調用JVM提供的接口來轉換成C類型的,就是通過調用env方法來完成轉換的。63 // obj代表調用的對象,相當于c++的this。當c函數需要改變調用對象成員變量時,可以通過操作這個對象來完成。64 // 編譯生成libHelloWorld.so65 // 在linux下執行如下命令來完成編譯工作: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.c69 // 在當前目錄生成libHelloWorld.so。注意一定需要包含Java的include目錄(請根據自己系統環境設定),因為Helloworld.c中包含了jni.h。70 // 另外一個值得注意的是在HelloWorld.java中我們LoadLibrary方法加載的是“HelloWorld”,可我們生成的Library卻是libHelloWorld。這是Linux的鏈接規定的,一個庫的必須要是:lib+庫名+.so。鏈接的時候只需要提供庫名就可以了。71 // 運行Java程序HelloWorld72 // 大功告成最后一步,驗證前面的成果的時刻到了:73 java HelloWorld74 // 如果你這步發生問題,如果這步你收到java.lang.UnsatisfiedLinkError異常,可以通過如下方式指明共享庫的路徑:75 java -Djava.library.path='.' HelloWorld76 // 當然還有其他的方式可以指明路徑請參考《在Linux平臺下使用JNI》。77 // 我們可以看到久違的“Hello world!”輸出了。View Code
試著去完成,自己生成了一份com_test_GetMsg.h頭文件,并完成test.c,生成libtest.so文件,JAVA調用SO文件時,屢次報:
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的。
注意,剛才引用的實例是JAVA調用SO,而我需要的是android調用SO,不然會頻繁上面錯誤。
原因有兩點:
1、JAVA和android的虛擬環境不一樣 2、Linux和android的系統庫文件不一樣
這樣導致了在Linux下通過JNI標準命名方式編譯的SO文件,在android是調用失敗的,原因是Linux和android的系統庫不一樣,而生產的SO跟生產環境庫文件有依賴關系,然后搭建了NDK和Cywin環境,然后生產的SO可以被android調用,
參考地址:http://zctya.blog.163.com/blog/static/1209178201181074018603/
那么SO文件就必須完全遵循JNI命名規則,方法名是這樣:
/* * Class: com_samples_jni_test * Method: GetMsg * Signature: ()V */JNIEXPORT jstring JNICALL Java_com_samples_jni_test_GetMsg(JNIEnv *, jobject);
通過NDK和Cywin生產libtest.so,android調用成功!
新聞熱點
疑難解答