搭建NDK环境 AndroidStudio3.5 Jni开发
- 前言
- 前期准备
- 正式开始Jni操作
- 1、新建一个Android Studio 工程demojniutil。新建一个JniUtil.java文件 、只声明,不实现
- 2、配置工程ndk路径、gradle.properties文件
- 3、新建一个jni文件夹,然后打开Android Studio的终端,cd到这个目录,然后javac命令生成java类的头文件
- 4、jni文件夹下新建Android.mk和Application.mk文件,同时新建c文件,用来实现3步骤的头文件的接口方法
- 5、将java类和实现java类的cpp源文件链接起来
- 6、配置app构建文件build.gradle
- 7、8、 9 运用ndk-build生成相应的so文件
- 10、MainActivity 使用
前言
因为项目需要,使用到了TEE,android平台上,使用TEE 来存储重要信息放到 TA里面,然后CA访问,最后android 和CA通过JNI 来交互。
使用到了CA(Client Application,运行在normal world)和TA(Trust Application,运行在secure world)
TA又分为用户TA(user ta)和静态TA(static ta),静态TA是和optee build在一起的
CA主要通过TEEC_UUID来调用TA
Java层调用库包:System.loadLibrary(“control_opt”); 引用native层函数
调用jni函数 Jni层:调用CA程序。TEE 的CA 和TA 已经跑通了,现在就需要android上层调用使用了,故此Jni开发开始,之前使用jni很少,这次在搭建NDK开发环境和使用Jni花了好多时间也走了好多弯路,今天写这篇给记录下来备份一下。
前期准备
如果你之前使用android studio 开发过就不要准备了,如果没有就做下下面操作吧:
- Android Studio3.5,配置Gradle ,Gradle 版本我选择的是:com.android.tools.build:gradle:3.5.2; 我选择这个版本解决一个坑,关系到cmake问题。
- 下载配置NDK,开发JNI 肯定需要NDK的,这是前提,我选择了NDK版本android-ndk-r14b;也是解决一个坑
- 安装配置JDK,Jdk至少要jdk7以上,我的是jdk8, version “1.8.0_121”;
正式开始Jni操作
整个流程大概是介个样的
- 新建一个Android Studio 工程JniHelloWorld。新建一个MyJni.java文件 、只声明,不实现
- 配置工程ndk路径、gradle.properties文件
- **新建一个jni文件夹,然后打开Android Studio的终端,cd到这个目录,然后javac命令生成java类的头文件 **
- jni文件夹下新建Android.mk和Application.mk文件,同时新建c文件,用来实现3步骤的头文件的接口方法
- 将java类和实现java类的cpp源文件链接起来
- 关联下载好的ndk包
- 配置app构建文件build.gradle
- 运用ndk-build生成相应的so文件
- 调用so文件 sourceSets{ //不配的话都会有一个默认值 可以指定哪些源文件(或文件夹下的源文件)要被编译,哪些源文件要被排除
main{
jni.srcDirs = [] //禁用as自动生成mk
//jniLibs.srcDirs=[“src/main/libs” ] //so包就去src/main/libs目录下找
//jniLibs.srcDirs=[‘libs’]
}
} - MainActivity 使用
1、新建一个Android Studio 工程demojniutil。新建一个JniUtil.java文件 、只声明,不实现
在这里我新建了一个JniUtil.java ,声明定义了几个方法,
//定义一个方法,该方法在C中实现
public native String getString(); //native关键字指示以原生形式实现的方法.向编译器告知实现在原生库中
public native int add(int i, int j);
native 关键字指示以原生形式实现的方法.向编译器告知实现在原生库
static {
System.loadLibrary("JniUtil");
}
System.loadLibrary正是需要导入的.so文件,so文件名全称是 libJniUtil.so
2、配置工程ndk路径、gradle.properties文件
我遇到过 executingexternalnativebuildforcmakexxxCMakeLists.tx… 提示,然后网上各种找,尝试过N种方法都不行,最后找到了适合我自己的方法。 我NDk选定版本为 android-ndk-r14b 然==classpath ‘com.android.tools.build:gradle:3.5.2’ ==
android.useDeprecatedNdk=true
后面不再支持了,我还是写上了。
3、新建一个jni文件夹,然后打开Android Studio的终端,cd到这个目录,然后javac命令生成java类的头文件
点击 main 右键 new -> JNI Folder ,就会在mian下面,新建了一个jni文件夹。
点击Terminal 栏输入 指令 > javac -encoding utf8 -h .\jni .\java\com\revo\demojniutil\JniUtil.java
javac -encoding utf8 -h .\jni .\java\com\revo\demojniutil\JniUtil.java
需要在 app/src/main 目录下输入该指令。最后会生成项目java文件对应的.h文件
com_revo_demojniutil_JniUtil.h
.h 文件名是自动生成的,包名 + 类名
javac -encoding utf8 -h .\jni .\java\com\revo\demojniutil\JniUtil.java
打开生成的.h文件,我们能看到 java类声明的方法
Java_com_revo_demojniutil_JniUtil_getString
Java_com_revo_demojniutil_JniUtil_add
接下来我们就需要在C文件是实现这些方法了
// An highlighted block
/*
* Class: com_revo_demojniutil_JniUtil
* Method: getString
* Signature: ()Ljava/lang/String;
*/
JNIEXPORT jstring JNICALL Java_com_revo_demojniutil_JniUtil_getString
(JNIEnv *, jobject);
/*
* Class: com_revo_demojniutil_JniUtil
* Method: add
* Signature: (II)I
*/
JNIEXPORT jint JNICALL Java_com_revo_demojniutil_JniUtil_add
(JNIEnv *, jobject, jint, jint);
4、jni文件夹下新建Android.mk和Application.mk文件,同时新建c文件,用来实现3步骤的头文件的接口方法
在jni文件夹下,new -> File -> Android.mk 和 Application.mk 。
New -> C/C++ Source File 新建一个C文件。JniLib.cpp
// Android.mk
LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
LOCAL_LDFLAGS := -Wl,--build-id
LOCAL_MODULE := JniUtil
LOCAL_SRC_FILES := \
JniUtil.c \
JniLib.cpp
MODULE_CPPFLAGS:= -std=c++11
LOCAL_LDLIBS += -llog
LOCAL_C_INCLUDES += $(LOCAL_PATH)/include
LOCAL_PROGUARD_ENABLED:= disabled
include $(BUILD_SHARED_LIBRARY)
// Application.mk
APP_MODULES := JniUtil
APP_ABI := all
// JniLib.cpp
//
// Created by Administrator on 2020/8/13.
//
#include <jni.h>
#include <com_revo_demojniutil_JniUtil.h>
#include <stdio.h>
#include <string.h>
#include<android/log.h>
/*
* Class: com_revo_demojniutil_JniUtil
* Method: getString
* Signature: ()Ljava/lang/String;
*/
extern "C" {
JNIEXPORT jstring JNICALL Java_com_revo_demojniutil_JniUtil_getString
(JNIEnv *env, jobject obj){
const char keyvalue[8] = {'s','t','v','e','l','z','s','x'};
return env->NewStringUTF(keyvalue);
}
}
/*
* Class: com_revo_demojniutil_JniUtil
* Method: add
* Signature: (II)I
*/
extern "C" {
JNIEXPORT jint JNICALL Java_com_revo_demojniutil_JniUtil_add
(JNIEnv *env, jobject obj, jint k, jint j){
return k + j;
}
}
==C++语言文件方法是向,加上 extern “C” { 包括否则要报错 ==
Cpp源码文件 JniLib.cpp 可以将 .h 的方法拷贝进去来实现,但是要记住一定要include 二个头文件。
#include <jni.h>
#include <com_revo_demojniutil_JniUtil.h>
5、将java类和实现java类的cpp源文件链接起来
那么java类和cpp 源文件怎么关联起来呢,在Java 目录右键 Link C++ Projiect with Gradle。
弹框 选在ndk-build, 在Project Path 选在项目jni文件下自己的Android.mk 文件。
操作后,会看到java文件的方法声明 有 C++的图标,C源文件有 Java图标。然后在build.gradle 会有
externalNativeBuild {
ndkBuild {
path file(‘src/main/jni/Android.mk’)
}
}
6、配置app构建文件build.gradle
配置build.gradle 选在编译和 jni so文件路径 导入名称等。
ndk 、 sourceSets 、task ndkBuild(type:Exec,description:‘Compile JNI source via NDK’) 三个节点配置
// An highlighted block
ndk{
moduleName "JniUtil"
ldLibs "log"//实现__android_log_print
abiFilters "armeabi","armeabi-v7a","arm64-v8a" //输出指定的三种abi体系下的so库"armeabi",只加载armabi架构(目录下)的so库,如果是别的架构,就会找不到
}
sourceSets{ //不配的话都会有一个默认值 可以指定哪些源文件(或文件夹下的源文件)要被编译,哪些源文件要被排除
main{
jni.srcDirs = [] //禁用as自动生成mk
//jniLibs.srcDirs=["src/main/libs" ] //so包就去src/main/libs目录下找
//jniLibs.srcDirs=['libs']
}
}
// An highlighted block
task ndkBuild(type:Exec,description:'Compile JNI source via NDK'){
commandLine "D:\\SDK\\sdk\\ndk\\android-ndk-r14b\\ndk-build.cmd",//配置ndk的路径
'NDK_PROJECT_PATH=build/intermediates/ndk',//ndk默认的生成so的文件
'NDK_LIBS_OUT=src/main/libs',//配置的我们想要生成的so文件所在的位置
'APP_BUILD_SCRIPT=src/main/jni/Android.mk',//指定项目以这个mk的方式
'NDK_APPLOCATION_MK=src/main/jni/Application.mk'//指定项目以这个mk的方式
}
7、8、 9 运用ndk-build生成相应的so文件
先配置一下ndk-build 环境,然后就可以运行ndk-build 生成 so文件了。
FIle -->Settings-> Tools - External Tools 添加 ndk-build 配置对应的参数。
Program: android-ndk-r14b\ndk-build.cmd 选择自己之前配置的ndk下面的 ndk-build.cmd
Workingdirectory: 选择到app\src\main 目录,可以点击后面的 Insert macros… 选择 ProjectFileDir 后就有 P r o j e c t F i l e D i r ProjectFileDir ProjectFileDir\app\src\main
配置好了ndk-build 环境,就可以在 java类 JniUtil.java 右键 External Tools 下 ndk-build。
然后就在下栏的run看到生成so文件的记录了,同时main目录下会多了一个libs文件夹,里面就是生成的so文件,如果有说明成功生成了。如果没有,就可以看看so文件run的记录,看看有没有报错。
10、MainActivity 使用
生成了so文件了,现在测试一下效果,在MainActivity 调用 jniUtil 类调用jniUitl.java 里面的add getstring 方法。log看到正常显示。
如果你留意的话,你会发现getstring log的结果有点不对,理论应该是stvelzsx,但是log后面还有乱码,这个就是C语言的问题了,就留给各位看官思考一下了,看看C语言的基础了。嘻嘻
点击查看代码 https://github.com/stvelzhang/demojniutil