JNI_Android项目中调用.so动态库

JNI_Android项目中调用.so动态库

2014年6月3日 JNI学习

參考:http://blog.sina.com.cn/s/blog_4298002e01013zk8.html

上一篇笔者介绍了怎样使用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();
}

以上代码声明三个本地方法。


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 __cplusplus
extern "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"

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是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文件



编译成功后会在project文件夹的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();
	}
}



执行项目,效果图例如以下:





posted @ 2017-04-24 12:40  zhchoutai  阅读(335)  评论(0编辑  收藏  举报