AndroidStudio3.5 NDK JNI开发

前言

因为项目需要,使用到了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 开发过就不要准备了,如果没有就做下下面操作吧:

  1. Android Studio3.5,配置Gradle ,Gradle 版本我选择的是:com.android.tools.build:gradle:3.5.2; 我选择这个版本解决一个坑,关系到cmake问题。
  2. 下载配置NDK,开发JNI 肯定需要NDK的,这是前提,我选择了NDK版本android-ndk-r14b;也是解决一个坑
  3. 安装配置JDK,Jdk至少要jdk7以上,我的是jdk8, version “1.8.0_121”;

正式开始Jni操作

整个流程大概是介个样的

  1. 新建一个Android Studio 工程JniHelloWorld。新建一个MyJni.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. 关联下载好的ndk包
  7. 配置app构建文件build.gradle
  8. 运用ndk-build生成相应的so文件
  9. 调用so文件 sourceSets{ //不配的话都会有一个默认值 可以指定哪些源文件(或文件夹下的源文件)要被编译,哪些源文件要被排除
    main{
    jni.srcDirs = [] //禁用as自动生成mk
    //jniLibs.srcDirs=[“src/main/libs” ] //so包就去src/main/libs目录下找
    //jniLibs.srcDirs=[‘libs’]
    }
    }
  10. 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

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值