Android項目中JNI技術生成並調用.so動態庫實現詳解


    生成 jni方式有兩種:一種是通過SWIG從C++代碼生成過度的java代碼;另一種是通過javah的方式從java代碼自動生成過度的C++代碼。兩種方式下的步驟流程正好相反。

第一種方式:由於需要配置SWIG環境,有點麻煩了,所以往往大家不采用這個途徑,參照博文http://my.oschina.net/liusicong/blog/314162

第二種方式:javah的方式則通過shell指令就可以完成整個流程,該過程大概包括以下步驟:

  1. 編寫 Java 代碼。我們將從編寫 Java 類開始,這些類執行三個任務:聲明將要調用的本機方法;裝入包含本機代碼的共享庫;然后調用該本機方法。
  2. 編譯 Java 代碼。在使用 Java 類之前,必須成功地將它們編譯成字節碼。

  3. 創建 C/C++ 頭文件。C/C++     頭文件將聲明想要調用的本機函數說明。然后,這個頭文件與 C/C++ 函數實現(請參閱步驟 4)一起來創建共享庫(請參閱步驟 5)。

  4. 編寫 C/C++ 代碼。這一步實現 C 或     C++ 源代碼文件中的函數。C/C++ 源文件必須包含步驟 3 中創建的頭文件。

  5. 創建共享庫文件。從步驟 4 中創建的 C 源代碼文件來創建共享庫文件。

  6. 運行 Java 程序。運行該代碼,並查看它是否有用。我們還將討論一些用於解決常見錯誤的技巧。

第二種方法的具體實現方式和例子如下:

1. 在Eclipse中創建項目:TestJNI

2. 新創建一個class:TestJNI.java

package com.wwj.jni;
 public class TestJNI {    
    public native boolean Init();    
    public native int Add(int x, int y);    
    public native void Destory();
}

以上代碼聲明三個本地方法。

3. 編譯JNI

找到Android項目中bin目錄下,會有classes文件夾,Eclipse自動為我們生成的字節碼文件就在這個目錄下。

我們在該路徑下,使用javah命令,生成我們想要得到的.h頭文件,如下圖所示:

執行javah -jni com.wwj.jni.TestJNI命令之后,會在classes目錄下生成頭文件:com_wwj_jni_TestJNI.h

將它復制到jni文件夾下,打開如下:

/* DO NOT EDIT THIS FILE - it is machine generated */
 #include <jni.h>
/* Header for class com_wwj_jni_TestJNI */
 #ifndef _Included_com_wwj_jni_TestJNI
 #define _Included_com_wwj_jni_TestJNI
 #ifdef __cplusplusextern "C" {
#endif
/*
 * Class:     com_wwj_jni_TestJNI
 * Method:    Init
 * Signature: ()Z
 */
 JNIEXPORT jboolean JNICALL Java_com_wwj_jni_TestJNI_Init
  (JNIEnv *, jobject);
/*
 * Class:     com_wwj_jni_TestJNI
 * Method:    Add
 * Signature: (II)I
 */
JNIEXPORT jint JNICALL Java_com_wwj_jni_TestJNI_Add
  (JNIEnv *, jobject, jint, jint);
/*
 * Class:     com_wwj_jni_TestJNI
 * Method:    Destory
 * Signature: ()V 
 */
 JNIEXPORT void JNICALL Java_com_wwj_jni_TestJNI_Destory
  (JNIEnv *, jobject);
#ifdef __cplusplus
}
#endif
#endif

以上代碼就是通過javah命令生成jni層代碼。

4. 使用C/C++實現JNI

在jni文件夾下,創建com_wwj_jni_TestJNI.h對應的cpp文件:com_wwj_jni_TestJNI.cpp

 

我們再添加兩個文件Add.h,Add.cpp,具體實現放在這兩個文件中來完成。

Add.h

#ifndef _TEST_JNI_ADD_H_
 #define _TEST_JNI_ADD_H_
    class CAdd {public:
         CAdd();    
        ~CAdd();    
        int Add(int x, int y);
    };
#endif

 

Add.cpp

#include "Add.h"
 CAdd::CAdd() {
 }
 CAdd::~CAdd() {
 }
 int CAdd::Add(int x, int y) {    
 return x + y;
}

com_wwj_jni_TestJNI.cpp的實現:

 

#include <stdio.h>
#include <stdlib.h>
#include "com_wwj_jni_TestJNI.h"
#include "Add.h"
Add *pCAdd = NULL;
JNIEXPORT jboolean JNICALL Java_com_wwj_jni_TestJNI_Init(JNIEnv *env,jobject obj) {   
   if (pCAdd == NULL) {
        pCAdd = new CAdd;
    }    
   return pCAdd != NULL;
}
JNIEXPORT jint JNICALL Java_com_wwj_jni_TestJNI_Add(JNIEnv *env, jobject obj,
        jint x, jint y) {    
        int res = -1;    
       if (pCAdd != NULL) {
          res = pCAdd->Add(x, y);
        }    
       return res;
}
JNIEXPORT void JNICALL Java_com_wwj_jni_TestJNI_Destory(JNIEnv *env, jobject obj)
{    if (pCAdd != NULL)
    {
        pCAdd = NULL;
    }
}

5. 創建mk文件,並使用ndk-build命令生成.so動態鏈接庫文件

在jni目錄下創建Android.mk文件如下:

LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
LOCAL_MODULE := TestJNI
LOCAL_SRC_FILES := com_wwj_jni_TestJNI.cpp
LOCAL_SRC_FILES += Add.cpp
include $(BUILD_SHARED_LIBRARY)

其中:

LOCAL_PATH是C/C++代碼所在目錄,也就是我們的jni目錄。

LOCAL_MODULE是要編譯的庫的名稱。編譯器會自動在前面加上lib,在后面加上.so。

LOCAL_SRC_FILES是要編譯的C/C++文件。

然后我還需要在Android項目根目錄下創建Application.mk文件:

APP_PROJECT_PATH := $(call my-dir)
APP_MODULES := TestJNI

寫完了這兩個mk文件,我們就可以用ndk來為我們生成相應的動態鏈接庫了。前提你需要下載NDK,並把NDK路徑配置到path環境變量中去,筆者配置的路徑是:D:\Cocos2dx\android-ndk-r9d,具體視個人情況而定。

進入Application.mk文件所在目錄,在命令行中使用ndk-build生成.so文件

編譯成功后會在工程目錄的libs/armeabi目錄下生成一個libTestJNI.so文件。

項目結構會變成如下:

6. 在Java中調用JNI

package com.wwj.jni;
import android.os.Bundle;
import android.widget.TextView;
import android.app.Activity;
public class TestJNIActivity extends Activity {    
private TextView textView;    
static {        // 加載動態庫
        System.loadLibrary("TestJNI");
    }
    @Override    
protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        textView = (TextView) findViewById(R.id.textview);
        TestJNI testJNI = new TestJNI();        // 調用native方法
        boolean init = testJNI.Init();        
       if (init == true) {            // 調用Add函數
            int sum = testJNI.Add(100, 150);
            textView.setText("你真是個" + sum);
        } else {
            textView.setText("你比二百五還要二百五");
        }
        testJNI.Destory();
    }
}

運行項目,效果圖如下:

按照以上步驟完全可以生成一個屬於自己的.so文件,且可正確調用和執行,這里補充一下兩點:

注意一:NDK的環境配置:

1,下載地址:

Android NDK r10e:

    32位:http://dl.google.com/android/ndk/android-ndk-r10e-windows-x86.exe

    64位:http://dl.google.com/android/ndk/android-ndk-r10e-windows-x86_64.exe

Android NDK r9d:

    32位:https://dl.google.com/android/ndk/android-ndk-r9d-windows-x86.zip

    64位: https://dl.google.com/android/ndk/android-ndk-r9d-windows-x86_64.zip

自己用的是ndk-r9d版本,以上地址需要翻牆,需要的對應版本留下郵箱就好。

2,配置PATH環境變量,cmd命令行輸入ndk-build,出現一下內容提示表示配置正確。

Android NDK: Could not find application project directory !
Android NDK: Please define the NDK_PROJECT_PATH variable to point to it.
D:\java\android-ndk-r9d\build/core/build-local.mk:148: *** Android NDK: Aborting
    .  Stop.

 

注意二:參考以上文章在最后輸入ndk-build時會提示以下錯誤:

此時修改一下Application.mk文件里的:

APP_PROJECT_PATH := $(call my-dir)/ 成 APP_PROJECT_PATH := $(call my-dir)/..

 

 

文章轉載於:http://www.cnblogs.com/sevenyuan/p/4202759.html


免責聲明!

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



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