Android进阶+实战android技术专栏Android知识

Android 生成.so文件

2016-09-20  本文已影响6567人  SheHuan

Android中,我们经常会用到.so文件,.so文件是一个C/C++的函数库,Android中调用.so文件都是通过jni的方式。Android系统中每一个CPU架构对应一个ABI,目前有以下几种:armeabi,armeabi-v7a,x86,mips,arm64-v8a,mips64,x86_64,所以我们可以生成对应ABI的.so文件。

接下来看如何生成我们的.so文件。

一、NDK环境搭建

1、下载NDK
下载链接:https://developer.android.com/ndk/downloads/index.html
当然没有梯子的话可以baidu找一下。PS:建议下载r9+的版本

2、下载完后解压,并将其根目录配置到Android Studio中,打开Project Structure(win快捷键:ctrl+alt+shfit+s),如下图:

配置ndk_1
在红框中配置你的ndk根目录,然后ok确认。

3、打开项目根目录的local.properties文件,如下:

配置ndk_2

可以看到ndk的目录,说明添加成功了。

4、打开项目根目录的gradle.properties文件,添加android.useDeprecatedNdk=true,如下:

配置ndk_3

到此ndk环境搭建完毕。

二、Java、C/C++代码编写

1、创建一个JniUtil类,以实现jni调用

public class JniUtil {
    static {
        //jniutil这个参数,可根据需要任意修改
        System.loadLibrary("jniutil");
    }

    //java调C/C++中的方法都需要用native声明且方法名必须和C/C++的方法名一样
    public native String test();
}

然后Make Project:

Make Project
完成后会生成对应的class文件:
JniUtil.class
2、根据JniUtil.class生成.h文件
打开Android Studio的Terminal,切换到项目的app/src/main目录下,执行命令:
javah -d jni -classpath 编译后的class文件的绝对路径

如下图:

jni命令

即可生成在app/src/main目录下自动创建一个包含.h文件的jni文件夹:


.h文件

我们不需要对这个.h文件做任何修改,其内容如下:

#include <jni.h>

#ifndef _Included_com_othershe_jnitest_JniUtil
#define _Included_com_othershe_jnitest_JniUtil
#ifdef __cplusplus
extern "C" {
#endif
JNIEXPORT jstring JNICALL Java_com_othershe_jnitest_JniUtil_test
  (JNIEnv *, jobject);

#ifdef __cplusplus
}
#endif
#endif

3、编写jniutil.c
这里的jniutil文件名需要和JniUtil类中System.loadLibrary("jniutil");的参数一致。
jniutil.c具体的编写可根据自己的业务实现,这里仅做测试:


#include "com_othershe_jnitest_JniUtil.h"

JNIEXPORT jstring JNICALL Java_com_othershe_jnitest_JniUtil_test
        (JNIEnv *env, jobject obj) {
    return (*env)->NewStringUTF(env, "jni调用成功");
}

在jniutil.c中我们需要导入上边的.h文件,然后实现具体的test方法。

4、配置项目app目录下的build.gradle文件

defaultConfig {
        applicationId "com.othershe.jnitest"
        minSdkVersion 14
        targetSdkVersion 24
        versionCode 1
        versionName "1.0"

        ndk {
            moduleName "jniutil"
            abiFilters 'armeabi', 'x86', 'armeabi-v7a'
        }
    }

其中ndk标签是新添加的,moduleName 的值同样为System.loadLibrary("jniutil");的参数。由于配置了abiFilters,则只会得到armeabi、x86、armeabi-v7a三种ABI对应的.so文件。

最后还需要在生成的jni文件夹下创建一个空的util.c文件,否则会有如下异常:


Exception

jni文件夹的结构如下:

jni目录结构

三、生成.so文件

简单的做下调用操作,运行项目看效果:

textView = (TextView) findViewById(R.id.test);
textView.setText(new JniUtil().test());
result

到这里就成功的实现了jni的调用。

说好的.so文件呢?看下图:


so文件

上边运行项目后就生成了对应ABI的.so文件。

四、使用生成的.so文件

上边的这些步骤只是帮助我们得到需要的.so文件,供自己的项目或给其它项目使用。所以使用阶段就简单多了:
首先在项目的app/src/main目录创建一个jniLibs文件夹,将生成的.so文件连同对应的所在的文件夹拷贝到创建的jniLibs中,如下图:

jniLibs

之前的的jni文件夹和build.gradle文件配置就不需要了。调用方法还和上边的一样。

PS:关于.so文件使用应该注意的问题可参考这篇文章:关于Android的.so文件你所需要知道的


测试代码

上一篇 下一篇

猜你喜欢

热点阅读