最近一直在练车,晒的是好黑好黑啊,好久没写博客了,趁着中午有时间,赶紧写一篇,写完就得骑着我的宝马电瓶车去驾校学车,累死哦。
为了满足as和ec的用户,我两种IDE都讲,好久没水了,装装X。
首先,得做好开发准备,下载好NDK包,as用户呢,可以在setting–>Android SDK里面找到,如下图:
可以看看自己的NDK是否安装了,如果没安装就勾选,并点击Apply就会自动下载了,下载的安装包会在sdk目录下看到,是一个叫ndk-bundle的文件夹,然后我们将这个路径设置到环境变量中,可以用ndk-build来测试一下是否已添加进环境变量中,ec的用户可以直接打开sdk manager去下载,生成的文件和上述情况一样。
好了,准备工作做好了。接下来就是开发了。。。。
Android Studio 用户:
首先来看看目录结构:
在app目录下新建了存储.c和mk文件,这些文件是为了生成so文件准备的,java目录下是调用的支持类JniTools.java。
好了,接下来分步骤讲解吧。跟着一步一步的就能实现啦!
第一步,新建一个调用类JniTools
package jni;
/**
* Created by wangqi on 2016/8/8.
*/
public class JniTools {
public native int add(int a,int b);
}
类中的本地方法是没有方法体的,并且用native修饰方法。
第二步,生成.h文件
首先打开Terminal,cd到jni目录的上一级,然后输入”包名.类名”的全路径,不然会找不到该类,要么就是so文件调用不了,输入javah jni.JniToos,接着就会在与jni同级目录中找到一个jni_JniTools.h的文件,这个文件名就是以全路径来命名的,你的调用类也必须这个全路径名,不然会报错,如图:
然后,我们来打开.h文件来看看:
/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class jni_JniTools */
#ifndef _Included_jni_JniTools
#define _Included_jni_JniTools
#ifdef __cplusplus
extern "C" {
#endif
/*
* Class: jni_JniTools
* Method: add
* Signature: (II)I
*/
JNIEXPORT jint JNICALL Java_jni_JniTools_add
(JNIEnv *, jobject, jint, jint);
#ifdef __cplusplus
}
#endif
#endif
可以看到我们熟知的方法add,前面加了好多不熟知的方法修饰,先不管他,这个方法似乎没有实现方法体,我们来实现它。
/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class jni_JniTools */
#ifndef _Included_jni_JniTools
#define _Included_jni_JniTools
#ifdef __cplusplus
extern "C" {
#endif
/*
* Class: jni_JniTools
* Method: add
* Signature: (II)I
*/
JNIEXPORT jint JNICALL Java_jni_JniTools_add
(JNIEnv *evn, jclass clazz, jint a, jint b) {
return a + b;
}
#ifdef __cplusplus
}
#endif
#endif
注意方法内部的参数默认是没有对象的,还有就是jobject最好改成jclass,有人在博客中提到这也会成为调用不了的一个因素,好了,方法体实现了,接着,将这个.h文件,我们强制命名为.c文件,然后在app的目录下新建一个jni目录,将这个.c文件复制到app目录下的jni文件下。
第三步,实现mk文件
接着,我们在app目录下的jni目录下新建一个mk文件
LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
LOCAL_MODULE := JniTools
LOCAL_SRC_FILES := jni_JniTools.c
include $(BUILD_SHARED_LIBRARY)
文件内容我们先关注LOCAL_MODULE ,这个后天跟着的是你加载的库名称,LOCAL_SRC_FILES 是你.c文件的文件名。
第四步,ndk-build命令生成so文件
我们cd到app目录下的jni目录下,然后输入命令ndk-build,你会看到:
E:\AS_workspace\ViewTools\app\jni>ndk-build
[arm64-v8a] Install : libJniTools.so => libs/arm64-v8a/libJniTools.so
[x86_64] Install : libJniTools.so => libs/x86_64/libJniTools.so
[mips64] Install : libJniTools.so => libs/mips64/libJniTools.so
[armeabi-v7a] Install : libJniTools.so => libs/armeabi-v7a/libJniTools.so
[armeabi] Install : libJniTools.so => libs/armeabi/libJniTools.so
[x86] Install : libJniTools.so => libs/x86/libJniTools.so
[mips] Install : libJniTools.so => libs/mips/libJniTools.so
这就是生成成功了,接着你就会看到app目录下多了一个libs文件和obj文件,打开libs文件,你会看到各种适应机型的so文件。
第五步,调用so里面的实现方法
public class MainActivity extends AppCompatActivity {
static {
System.loadLibrary("JniTools");
}
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
JniTools jniTools = new JniTools();
Log.i("jni", jniTools.add(10, 20) + "--------------------");
}
首先得先加载类库,这个库名称就是mk文件里面LOCAL_MODULE后面所对应的名称,接着,我们调用jniTools里面的add方法,这个方法会自动去寻找so文件内实现的方法体,完成调用。
成功运行截图
eclipse用户:
实现步骤和上面一模一样,我先展示一下结构图。
首先在包下面新建一个实现类,然后cd到src目录下,敲命令javah “包名.类名”生成的.h文件打开实现里面的方法,和上面一样,接着在JNI项目文件夹下新建一个jni目录,将.h文件放进去,并改成.c文件,然后新建mk文件,和上面一样,然后cd到jni目录下,敲入ndk-build,会在JNI目录下多出一个libs文件夹和一个obj文件夹,接着我们就在MainActivity中去调用。
package com.example.jni;
import android.app.Activity;
import android.os.Bundle;
import android.widget.TextView;
public class MainActivity extends Activity {
static {
System.loadLibrary("jnitest");
}
private TextView tv;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
tv = (TextView) findViewById(R.id.add);
jnitest jnitest = new jnitest();
int a = jnitest.add(10, 20);
tv.setText(a + "");
}
}
成功图样
好了,结束了,要是实现中有不对的地方,和我说说,我来看看。
对了,as有好几种项目视图,我一般用Android和Project这两种,有些刚用as的用户,可能因为不适应,新建libs文件夹后倒入jar文件,gradle后发现调用不了里面的方法,我感觉可能是因为你libs文件夹可能放错位置了,记住,无论是eclipse还是Android Studio,libs文件夹一定是与src文件夹是在同一级目录,发现不是的话那就是新建错了。
好了,不说了,要去学车了,争取当个老司机。