AndroidStudio jni开发入门及打包so库和jar包

AndroidStudio jni开发入门及打包so库和jar包

配置ndk开发环境

  1. 下载NDK,LLDB,CMake工具包
  2. 配置系统环境变量

在原有项目中进行jni开发环境配置

Androidstudio2.0之后新建的项目可直接生成cpp项目,所有详细说下在原有项目中进行jni开发环境配置:

  1. 在src/main文件夹下建一个cpp文件夹(这里存放jni代码)
  2. 在app文件夹下建一个CMakeLists.txt文件,CMake是一种跨平台编译工具
代码如下:
cmake_minimum_required(VERSION 3.4.1)

# add_library()可添加多个,有多少个cpp文件可添加多少个
add_library( 
 # 对应cpp的文件名
 jni-array

 # Sets the library as a shared library.
 SHARED

 # 对应cpp的文件名地址
 src/main/cpp/jni-array.cpp )


find_library( # Sets the name of the path variable.
  log-lib

  # Specifies the name of the NDK library that
  # you want CMake to locate.
  log )

#
target_link_libraries( # Specifies the target library.
   jni-array

   # Links the target library to the log library
   # included in the NDK.
   ${log-lib} )
  1. 配置app下的gradle文件
代码
android {
    compileSdkVersion 28
    buildToolsVersion "28.0.2"
    defaultConfig {
	    applicationId "com.jiax.myapplication"
	    minSdkVersion 15
	    targetSdkVersion 28
	    versionCode 1
	    versionName "1.0"
	    testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner"
	    externalNativeBuild {
	    	cmake {
				//c++的版本
	    		cppFlags "-std=c++14"
				abiFilters 'armeabi-v7a','x86'
	    	}
    	}
		ndk{
            moduleName "jniTest"
            abiFilters "armeabi-v7a","x86"
    	}
	}
	
    buildTypes {
	    release {
		    minifyEnabled false
		    proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro'
	    }
    }
	//关联CMakeLists.txt文件
    externalNativeBuild {
	    cmake {
	    	path "CMakeLists.txt"
	    }
    }
}

编写C层代码

在cpp文件夹中建一个jni-array.cpp文件

####代码####

#include 
/*JNIEXPORT和JNICALL这两个宏(被定义在jni.h)确保这个函数在本地库外可见,并且C编译器会进行正确的调用转换。*/
extern "C"
/*
 *数组求和	*/
JNIEXPORT jint JNICALL
Java_com_jiax_yugang_java_SimpleJniUtil_sumArray(JNIEnv *env, jclass type, jintArray arr_) {
    /*jint *arr = env->GetIntArrayElements(arr_, NULL);
    env->ReleaseIntArrayElements(arr_, arr, 0);*/
    jsize len = env->GetArrayLength(arr_);
    int buf[len];
    int sum = 0;
    //将基本类型数组某一区域复制到缓冲区中的一组函数。
    env->GetIntArrayRegion(arr_, 0, len, buf);
    for (int i = 0; i < len; i++) {
        sum += buf[i];
    }

    return sum;
}

打包so库

1.在CMakeLists.txt文件中添加so库输出代码:

set(CMAKE_LIBRARY_OUTPUT_DIRECTORY ${PROJECT_SOURCE_DIR}/../jniLibs/${ANDROID_ABI})


在Build->Make Porject后台会在项目文件夹下生成一个jniLibs文件夹,里面就是打包好的so库

编写java代码

拿到so库时,要建立相关的java类对外提供api,包名要跟so库的方法名里的包名一致,方法名也必须一致才能成功调用so库中的方法
1.在Android项目下建一个module,包名必须跟之前生成so库里方法名的包名一致


2.编写jni代码,类名、方法名必须跟之前生成so库里方法名的类名、方法名一致
代码

public class SimpleJniUtil {
	//编写native方法
    public static native int doubleData(int data);
    public static native int sumArray(int[] arr);
    public static native int[] arrayAddTen(int[] arr);
    public static native int[][] init2DArray(int size);
    public native String getText();

    static {
        System.loadLibrary("jni-array");//导入so库
    }

}

3.MakeProject,生成.class文件

打包对外提供API的jar包

1.在module下的gradle文件中配置打包任务
代码如下

task makeJar(type: Copy) {
    delete 'build/libs/mylibrary.jar' //删除已经存在的jar包
    from('build/intermediates/bundles/release/')//从该目录下加载要打包的文件
    into('build/libs/')//jar包的保存目录
    include('classes.jar')//设置过滤,只打包classes文件
    rename('classes.jar', 'mylibrary.jar')//重命名,mylibrary.jar 根据自己的需求设置
}
makeJar.dependsOn(build)

2.打包

在工具栏切换到makejar任务运行即可打包,或在gradle栏mylibrary/other中找到makejar任务双击打包,打包后的jar会生成在你指定的build/libs/目录下

你可能感兴趣的:(ndk)