qileilove

          blog已經(jīng)轉(zhuǎn)移至github,大家請?jiān)L問 http://qaseven.github.io/

          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í)專欄

          <2014年11月>
          2627282930311
          2345678
          9101112131415
          16171819202122
          23242526272829
          30123456

          導(dǎo)航

          統(tǒng)計(jì)

          常用鏈接

          留言簿(55)

          隨筆分類

          隨筆檔案

          文章分類

          文章檔案

          搜索

          最新評論

          閱讀排行榜

          評論排行榜

          主站蜘蛛池模板: 苍溪县| 景德镇市| 松江区| 仪征市| 鲜城| 蕉岭县| 北宁市| 扬中市| 随州市| 普格县| 临武县| 泸西县| 庆阳市| 朔州市| 阿克陶县| 颍上县| 二连浩特市| 荔波县| 潜江市| 盘山县| 历史| 九龙县| 齐齐哈尔市| 延庆县| 金川县| 奉化市| 渭南市| 苏尼特左旗| 永昌县| 连平县| 屯门区| 南汇区| 博罗县| 遵化市| 金坛市| 卢氏县| 县级市| 习水县| 凤庆县| 合山市| 太白县|