JAVA調用C語言寫的SO文件


  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     static
14     {
15         System.loadLibrary("HelloWorld");
16     }
17 }
18 // 注意print方法的聲明,關鍵字native表明該方法是一個原生代碼實現的。另外注意static代碼段的System.loadLibrary調用,這段代碼表示在程序加載的時候,自動加載libHelloWorld.so庫。
19 // 編譯HelloWorld.java
20 // 在命令行中運行如下命令:
21 javac HelloWorld.java
22 // 在當前文件夾編譯生成HelloWorld.class。
23 // 生成HelloWorld.h
24 // 在命令行中運行如下命令:
25 javah -jni HelloWorld
26 // 在當前文件夾中會生成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_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 // 該文件中包含了一個函數Java_HelloWorld_print的聲明。這里面包含兩個參數,非常重要,后面講實現的時候會講到。
49 // 實現HelloWorld.c
50 // 創建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的兩個參數,本例比較簡單,不需要用到這兩個參數。但是這兩個參數在JNI中非常重要。
62 // env代表java虛擬機環境,Java傳過來的參數和c有很大的不同,需要調用JVM提供的接口來轉換成C類型的,就是通過調用env方法來完成轉換的。
63 // obj代表調用的對象,相當於c++的this。當c函數需要改變調用對象成員變量時,可以通過操作這個對象來完成。
64 // 編譯生成libHelloWorld.so
65 // 在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.c
69 // 在當前目錄生成libHelloWorld.so。注意一定需要包含Java的include目錄(請根據自己系統環境設定),因為Helloworld.c中包含了jni.h。
70 // 另外一個值得注意的是在HelloWorld.java中我們LoadLibrary方法加載的是“HelloWorld”,可我們生成的Library卻是libHelloWorld。這是Linux的鏈接規定的,一個庫的必須要是:lib+庫名+.so。鏈接的時候只需要提供庫名就可以了。
71 // 運行Java程序HelloWorld
72 // 大功告成最后一步,驗證前面的成果的時刻到了:
73 java HelloWorld
74 // 如果你這步發生問題,如果這步你收到java.lang.UnsatisfiedLinkError異常,可以通過如下方式指明共享庫的路徑:
75 java -Djava.library.path='.' HelloWorld
76 // 當然還有其他的方式可以指明路徑請參考《在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調用成功!


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM