JNI_Android项目中调用.so动态库
2014年6月3日 JNI学习
上一篇笔者介绍了如何使用Java代码调用DLL动态库中的C/C++方法,似乎已经是很久以前的做法了,遇到的错误笔者还未找到解决方案,但动态库着实是找到的,只是无法调用相应的方法。本篇博客来介绍一下如何在Android项目当中使用NDK生成.so动态链接库,并在程序中使用。
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();
- }
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文件夹下,打开如下:
-
- #include <jni.h>
-
-
- #ifndef _Included_com_wwj_jni_TestJNI
- #define _Included_com_wwj_jni_TestJNI
- #ifdef __cplusplus
- extern "C" {
- #endif
-
-
-
-
-
- JNIEXPORT jboolean JNICALL Java_com_wwj_jni_TestJNI_Init
- (JNIEnv *, jobject);
-
-
-
-
-
-
- JNIEXPORT jint JNICALL Java_com_wwj_jni_TestJNI_Add
- (JNIEnv *, jobject, jint, jint);
-
-
-
-
-
-
- JNIEXPORT void JNICALL Java_com_wwj_jni_TestJNI_Destory
- (JNIEnv *, jobject);
-
- #ifdef __cplusplus
- }
- #endif
- #endif
/* 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
#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;
- }
#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"
-
- CAdd *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;
- }
- }
#include <stdio.h>#include <stdlib.h>#include "com_wwj_jni_TestJNI.h"#include "Add.h"CAdd *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 := $(call my-dir)include $(CLEAR_VARS)LOCAL_MODULE := TestJNILOCAL_SRC_FILES := com_wwj_jni_TestJNI.cppLOCAL_SRC_FILES += Add.cppinclude $(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
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();
-
- boolean init = testJNI.Init();
- if (init == true) {
-
- int sum = testJNI.Add(100, 150);
- textView.setText("你真是个" + sum);
- } else {
- textView.setText("你比二百五还要二百五");
- }
- testJNI.Destory();
- }
- }
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(); }}
运行项目,效果图如下: