当使用第三方提供的jni库或不打算开源,方法一就不适用了。这里实现一个jni动态库与应用工程分离实现的方法.
一 打开Android Studio, 菜单”Tools” – > “Android” –> “SDK Manager”在下图里选择安装”ndk”, 如已安装则跳过这步骤.
Ndk安装完成后,会默认安装在”~/Android/Sdk/ndk-bundle/”目录里.
二 因编译jni动态库时需要使用ndk里的工具,所以把ndk的所在目录加入系统环境”PATH”里,以便使用ndk里的工具.
先确定ndk的绝对路径,如我的在”/home/jk/Android/Sdk/ndk-bundle/”
修改”~/.bash_profile”文件,在文件增加一行,内容”export PATH=/home/jk/Android/Sdk/ndk-bundle/:$PATH”
完成后,注消系统后重新登录,以便系统环境变量生效.
三 编写jni库的代码.
在目录”com/jk/”下编写了一个Hello.java源文件, 内容:
package com.jk;
public class Hello {
public native int getStringLen(String str); //返回字符串长度
public native int add(int a, int b); //加法函数
}
在终端执行”javah com.jk.Hello”产生jni头文件”com_jk_Hello.h”
然后把头文件里的函数在一个c文件里实现.
/* com_jk_Hello.c */
#include <jni.h>
#include <string.h>
#include "com_jk_Hello.h"
JNIEXPORT jint JNICALL Java_com_jk_Hello_getStringLen
(JNIEnv *env, jobject obj, jstring str)
{
const char *data = (*env)->GetStringUTFChars(env, str, 0);
return strlen(data);
}
JNIEXPORT jint JNICALL Java_com_jk_Hello_add
(JNIEnv *env, jobject obj, jint a, jint b)
{
return a+b;
}
四 编译jni代码。 编译Android上的jni需调用ndk提供的工具,需要编写Android.mk文件描述编译目标.
/* Android.mk */
LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
LOCAL_LDLIBS := -L$(SYSROOT)/usr/lib -llog
LOCAL_PRELINK_MODULE := false
LOCAL_MODULE := myjni
LOCAL_SRC_FILES := com_jk_Hello.c
include $(BUILD_SHARED_LIBRARY)
编译出的动态库名为: libmyjni.so
同时还需要编写一个App.mk描述jni的应用工程.
/* App.mk */
APP_BUILD_SCRIPT := Android.mk
上面两个文件编写好后,就可以在终端上执行编译命令:
ndk-build NDK_PROJECT_PATH=. NDK_APPLICATION_MK=App.mk
命令执行完成后,编译出来的jni动态库就在当前目录的libs目录里,而且编译出的jni动态库有多种平台的版本, 分别有以平台命名的子目录里.
五 在Android工程里使用jni动态库.
随便创建一个Android工程后,把第四步编译出来的libs整个目录复制到Android工程源目录的app子目录里,替换里面的libs目录.
然后在Android Studio的切换成Project工程资源窗口就可以看到jni动态库了,在如下图的界面里切换:
查看到的动态库,如图:
接着写调用jni的java类。
首先新建一个包, 包名必须与第三步所作的一致。我的就必须创建一个”com.jk”包, 包里再创建Hello.java源文件, 如图:
Hello.java源文件里的内容:
/* Hello.java */
package com.jk;
public class Hello {
static {
System.loadLibrary("myjni");
}
public native int getStringLen(String str); //返回字符串长度
public native int add(int a, int b); //加法函数
}
接着还需要修改工程app目录下的build.gradle文件, 让工程把我们的jni动态库打包进工程最终生成的apk里. 从文件的第20行新增下面内容:
sourceSets.main {
jni.srcDirs = []
jniLibs.srcDir 'libs'
}
task nativeLibsToJar(type: Zip, description: "create a jar archive of the native libs") {
destinationDir file("$projectDir/libs")
baseName "libmyjni"
extension "jar"
from fileTree(dir: "libs", include: "**/*.so")
into "lib"
}
最后就可以工程里调用jni动态库的功能了, 如在Activity窗口创建时调用:
/* MainAcitivty.java */
package com.example.jk.testmyjni;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import com.jk.Hello;
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
//调用jni动态库功能
Hello h = new Hello();
System.out.printf("strlen = %d\n", h.getStringLen("hello_world"));
System.out.printf("33 + 44 = %d\n", h.add(33, 44));
}
}