[AOSP] [JNI] [Android] AOSP中使用JNI
一. 简要
在Android中,JNI主要用于实现一些性能较高的功能,如图像处理、音频处理、视频处理等。同时,JNI也可以用于实现一些特殊的功能,如与硬件交互、与系统服务交互等。
二. Java层
在某个对象中添加如下代码,例如我在/frameworks/base/services/core/java/com/android/server/keepalive/KeepAliveManagerService.java中去使用native方法
public class KeepAliveManagerService extends IKeepAliveManager.Stub {private static final String TAG = "KeepAliveManagerService leilei";private final Context mContext;private final ActivityManagerService mActivityManagerService;private WindowManagerService mWindowManagerService;private boolean mSystemReady = false;**public static native int resumeNative(int v);****public static native int pauseNative(int v);****public static native String stopNative(String v);**// 应用保活方法,简单的调用了三个native方法:resumeNative,pauseNative,stopNative@Overridepublic boolean keepAliveApplicationByPackage(String packageName) throws RemoteException {Log.d(TAG, "keepAliveApplicationByPackage.packageName:"+packageName);if (TextUtils.isEmpty(packageName) || mActivityManagerService == null|| mContext == null || !mSystemReady){return false;}int s = resumeNative(1);int s1 = pauseNative(2);String s2 = stopNative("leilei");Log.d(TAG, "leilei keepAliveApplicationByPackage: onResumeNative:"+s);Log.d(TAG, "leilei keepAliveApplicationByPackage: onPauseNative:"+s1);Log.d(TAG, "leilei keepAliveApplicationByPackage: stopNative:"+s2);int curCallingUid = Binder.getCallingUid();return keepAliveApplicationByPackage(packageName,curCallingUid);}
}
至此java层的代码就写好了
三. C/C++层
-
JNI文件创建
因为我写jni方法是需要在我的service对象里使用,所以我frameworks/base/services/core/jni/目录下创建为需要的文件,如下代码所示
#include <string> #include <dlfcn.h> #include <pthread.h> #include <chrono> #include <thread> #include <jni.h> #include <nativehelper/JNIHelp.h> #include <android/binder_manager.h> #include <android/binder_stability.h> #include <android/hidl/manager/1.2/IServiceManager.h> #include <binder/IServiceManager.h> #include <hidl/HidlTransportSupport.h> #include <incremental_service.h> #include <memtrackproxy/MemtrackProxy.h> #include <schedulerservice/SchedulingPolicyService.h> #include <sensorservicehidl/SensorManager.h> #include <stats/StatsAidl.h> #include <stats/StatsHal.h> #include <bionic/malloc.h> #include <bionic/reserved_signals.h> #include <android-base/properties.h> #include <utils/Log.h> #include <utils/misc.h> #include <utils/AndroidThreads.h> #ifdef LOG_TAG #undef LOG_TAG #define LOG_TAG "leilei" #endif// jni静态注册 /*extern "C" jstring Java_com_android_server_keepalive_KeepAliveManagerService_onResumeNative(JNIEnv *env, jclass thiz, jlong value) {// 进行本地处理,生成返回值std::string hello = "Hello from C++";jstring result = env->NewStringUTF(hello.c_str());return result; }*/// jni动态注册 namespace android { static jint pauseNative(JNIEnv* env, jobject thiz, jint value){ALOGD("The leilei message is onPauseNative %d:",value);return value; }static jint resumeNative(JNIEnv *env, jobject thiz, jint value){ALOGD("The leilei message is onResumeNative %d:",value);return value; }static jstring stopNative(JNIEnv *env, jobject thiz,jstring value){const char* ptr = env->GetStringUTFChars(value, NULL);ALOGD("The leilei message is stopNative %s:",ptr);return value; }// 对应native的方法注册表 static const JNINativeMethod gKeepAliveManagerMethods[] = {/* name, signature, funcPtr */{"pauseNative","(I)I",(void *)pauseNative},{"resumeNative","(I)I",(void *)resumeNative},{"stopNative","(Ljava/lang/String;)Ljava/lang/String;",(void*) stopNative}, };// 自己实现一个跟jniRegisterNativeMethods一样的功能 int registerNativeMethods(JNIEnv *env,std::string name,const JNINativeMethod *methods) {// 反射拿到java对象jclass klass = env->FindClass(name.c_str());if (klass == NULL) {return -1;}// 第一个参数:反射拿到的对象// 第二个参数:类中的native方法--注册表// 第三个参数:native方法对象的个数return env->RegisterNatives(klass, methods,3); }// onload.cpp中调用了JNI_OnLoad,然后调用了register_android_server_KeepAliveManager进行注册 // jniRegisterNativeMethods对RegisterNatives封装了,所以可以很方便的使用,我们手动来实现一下 // JNI_OnLoad是jni.h中的对象,只有调用JNI_OnLoad和RegisterNatives才是动态注册 int register_android_server_KeepAliveManager(JNIEnv* env) {// return jniRegisterNativeMethods(env, "com/android/server/keepalive/KeepAliveManagerService",gKeepAliveManagerMethods, NELEM(gKeepAliveManagerMethods));return registerNativeMethods(env,"com/android/server/keepalive/KeepAliveManagerService",gKeepAliveManagerMethods); } };
文件名称必须规范:com.android.server.keepalive.KeepAliveManagerService.cpp,由包名+类名组成,然后实现对应上层的native方法即可,方法名称最好相同(也可以不同,只要在注册函数的第一个参数中对应起来就行),例如下方法,三个参数中,前两个参数必须有,而且不能变—>JNIEnv和jobject:
static jint pauseNative(JNIEnv* env, jobject thiz, jint value){... }
然后就是开始动态注册jni方法,如下代码所示,由于aosp已经封装好了
jniRegisterNativeMethods
方法可以直接使用来注册jni方法,但是为了更深刻的理解,我们手动来实现registerNativeMethods
// native方法注册表 static const JNINativeMethod gKeepAliveManagerMethods[] = {/* name, signature, funcPtr */{"pauseNative","(I)I",(void *)pauseNative},{"resumeNative","(I)I",(void *)resumeNative},{"stopNative","(Ljava/lang/String;)Ljava/lang/String;",(void*) stopNative}, };// 自己实现一个跟register_android_server_KeepAliveManager一样的功能 int registerNativeMethods(JNIEnv *env,std::string name,const JNINativeMethod *methods) {jclass klass = env->FindClass(name.c_str());if (klass == NULL) {return -1;}// 第一个参数:反射拿到的对象// 第二个参数:类中的native方法--注册表// 第三个参数:native方法对象的个数return env->RegisterNatives(klass, methods,3); }int register_android_server_KeepAliveManager(JNIEnv* env) {// return jniRegisterNativeMethods(env, "com/android/server/keepalive/KeepAliveManagerService",gKeepAliveManagerMethods, NELEM(gKeepAliveManagerMethods));return registerNativeMethods(env,"com/android/server/keepalive/KeepAliveManagerService",gKeepAliveManagerMethods); }
先实现一个native方法注册表,代表需要对应java层native方法,返回的对象是
JNINativeMethod
,该对象属于jni.h里的,结构如下typedef struct {const char* name;const char* signature;void* fnPtr; } JNINativeMethod;
第一个参数对应了java native方法的名称,第二个参数代表native方法里面的参数和返回值,第三个参数代表jni方法
回到
registerNativeMethods
方法,主要就是通过env的FindClass反射获取Java对象,然后通过RegisterNatives(klass, methods,3);
进行注册即可,第二个参数就是native方法gKeepAliveManagerMethods
注册表**思考一下:**既然需要注册jni,那么调用
register_android_server_KeepAliveManager
函数的注册的入口又在哪里?下文分析 -
JNI文件引入和注册流程
-
android.bp引入编译
需要让我们的jni文件参与编译,需要在frameworks/base/services/core/jni/Android.bp中添加该文件,如下所示
cc_library_static {name: "libservices.core",defaults: ["libservices.core-libs"],cpp_std: "c++2a",cflags: ["-Wall","-Werror","-Wno-unused-parameter","-Wthread-safety","-DEGL_EGLEXT_PROTOTYPES","-DGL_GLEXT_PROTOTYPES",],srcs: ["BroadcastRadio/JavaRef.cpp","BroadcastRadio/NativeCallbackThread.cpp","BroadcastRadio/BroadcastRadioService.cpp","BroadcastRadio/Tuner.cpp","BroadcastRadio/TunerCallback.cpp","BroadcastRadio/convert.cpp","BroadcastRadio/regions.cpp","gnss/GnssConfiguration.cpp","gnss/GnssMeasurement.cpp","gnss/GnssMeasurementCallback.cpp","gnss/Utils.cpp","stats/SurfaceFlingerPuller.cpp",**"com.android.server.keepalive.KeepAliveManagerService.cpp",**include_dirs: ["frameworks/base/libs","frameworks/native/services","system/gatekeeper/include","system/memory/libmeminfo/include",],header_libs: ["bionic_libc_platform_headers",], }
在此模块添加**“com.android.server.keepalive.KeepAliveManagerService.cpp”**,即可,模块名为
libservices.core
,会生成对应的so库 -
JNI注册入口声明
在frameworks中,上文分析了如何调用jni注册native方法,但是调用注册的入口在哪里?就是通过frameworks/base/services/core/jni/onload.cpp文件进行调用的,需要在此文件中声明我们的注册入口,如下代码所示
namespace android { int register_android_server_BatteryStatsService(JNIEnv* env); int register_android_server_ConsumerIrService(JNIEnv *env); int register_android_server_InputManager(JNIEnv* env); **int register_android_server_KeepAliveManager(JNIEnv* env);** int register_android_server_LightsService(JNIEnv* env); int register_android_server_PowerManagerService(JNIEnv* env); int register_android_server_PowerStatsService(JNIEnv* env); int register_android_server_HintManagerService(JNIEnv* env); int register_android_server_storage_AppFuse(JNIEnv* env); } extern "C" jint JNI_OnLoad(JavaVM* vm, void* /* reserved */) {JNIEnv* env = NULL;jint result = -1;if (vm->GetEnv((void**) &env, JNI_VERSION_1_4) != JNI_OK) {ALOGE("GetEnv failed!");return result;}ALOG_ASSERT(env, "Could not retrieve the env!");register_android_server_broadcastradio_BroadcastRadioService(env);register_android_server_broadcastradio_Tuner(vm, env);register_android_server_PowerManagerService(env);register_android_server_PowerStatsService(env);register_android_server_HintManagerService(env);register_android_server_SerialService(env);register_android_server_InputManager(env);**register_android_server_KeepAliveManager(env);}** }
只需要在
namespace android
中声明注册入口函数**register_android_server_KeepAliveManager
**,此函数在我们创建的jni文件中会实现。然后在
JNI_OnLoad
函数中添加**register_android_server_KeepAliveManager(env);
**,目的是为了方法可以被正确调用,以及传递了env对象(jni里的东西),再来看一遍我创建的jni文件,frameworks/base/services/core/jni/com.android.server.keepalive.KeepAliveManagerService.cpp... int **register_android_server_KeepAliveManager**(JNIEnv* env) {// return jniRegisterNativeMethods(env, "com/android/server/keepalive/KeepAliveManagerService",gKeepAliveManagerMethods, NELEM(gKeepAliveManagerMethods));return registerNativeMethods(env,"com/android/server/keepalive/KeepAliveManagerService",gKeepAliveManagerMethods); } };
**思考一下:**为什么需要在
extern "C" jint JNI_OnLoad(JavaVM* vm, void* /* reserved */)
方法中调用jni注册逻辑呢?这涉及到jni动态注册原理了,后面再分析,jni分为静态注册和动态注册 -
JNI注册表分析
Andoird 中使用了一种不同传统Java JNI的方式来定义其native的函数。其中很重要区别是Andorid使用了一种Java 和 C 函数的映射表数组,并在其中描述了函数的参数和返回值。这个数组的类型是JNINativeMethod——该结构体位于
jni.h
中,结构如下typedef struct {const char* name;const char* signature;void* fnPtr; } JNINativeMethod;
三个参数代表着:native方法名称,签名—用字符串是描述了Java中函数的参数和返回值,jni函数对象-指向了java的native方法
具体用法如下
static const JNINativeMethod gKeepAliveManagerMethods[] = {/* name, signature, funcPtr */{"pauseNative","(I)I",(void *)pauseNative},{"resumeNative","(I)I",(void *)resumeNative},{"stopNative","(Ljava/lang/String;)Ljava/lang/String;",(void*) stopNative}, };
第三个参数前面必须带有(void *),这里主要分析第二个参数,()代表native方法的参数,()外面部分代表着返回值,I代表着java的int,jni的jint,具体如下
字符 c/c++类型 Java类型 V void void Z jboolean boolean I jint int J jlong long D jdouble double F jfloat float B jbyte byte C jchar char S jshort short 以上都是基本数据类,如果是数组,则用[代表,如整型数值
[I
来表示,具体如下名称 c/c++类型 Java类型 [I jintArray int[] [F jfloatArray float[] [B jbyteArray byte[] [C jcharArray char[] [S jshortArray short[] [D jdoubleArray double[] [J jlongArray long[] [Z jbooleanArray boolean[] 那如果native参数中是对象呢,需要用如下方法表示—参数解释:
// 参数解释 "()" 中的字符表示参数,小括号后面的则代表返回值。 "()V" 就表示native void Fun(); "(II)V" 表示native void Func(int a, int b);参数是俩个整型。 "(Ljava/lang/String;)Ljava/lang/String;" 就表示native Sting Func(String value);
所以如果要用对象作为参数或者返回值,需要在前面加个”L”,中间是用”/" 隔开,后面跟包名和类名,以及分号即可。如果是对象数组,则在前面加个
[
即可
-
相关文章:
[AOSP] [JNI] [Android] AOSP中使用JNI
一. 简要 🍎 JNI是Java Native Interface的缩写,它提供了若干的API实现了Java和其他语言的通信(主要是C&C)。从Java1.1开始,JNI标准成为java平台的一部分,它允许Java代码和其他语言写的代码进行交互。J…...
GEE案例——如何使用长时序影像实现多波段图像加载(不同层土壤湿度)
简介: 在GEE中实现时序图像的加载主要的目的是查看影像波段或者指数的变化,这里我们使用的主要是加载常规的4个波段,然后添加一个复合波段,复合波段主要的是求4个波段的平均值,然后再次加入到原有的4个波段的时序图中。这里面主要的技术难点一个是图表的设定,另外一个就…...

Cloudflare进阶技巧:缓存利用最大化
1. 引言 cloudflare我想你应该知道是什么,一家真正意义上免费无限量的CDN,至今未曾有哥们喷它的。当然,在国内的速度确实比较一般,不过这也不能怪它。 CDN最大的特色,我想就是它的缓存功能,达到防攻击&am…...

想要精通算法和SQL的成长之路 - 二叉树的判断问题(子树判断 | 对称性 | 一致性判断)
想要精通算法和SQL的成长之路 - 二叉树的判断问题 前言一. 相同的树二. 对称二叉树三. 判断子树 前言 想要精通算法和SQL的成长之路 - 系列导航 一. 相同的树 原题链接 这题目典型的递归题: 如果两个节点都是null,我们返回true。如果两个节点一个nul…...
(零)如何做机器视觉项目
文章目录 1 项目的前期准备1.1 从5个方面初步分析客户需求1.2 方案评估与验证1.3 签订合同 2 项目规划2.1 定义客户端的详细需求2.2 制定项目管理计划2.3 方案评审 3 详细设计3.1 硬件设备的选择与环境搭建3.2 软件开发平台与开发工具的选择3.3 机器视觉系统的整体框架与开发流…...

【Leetcode】滑动窗口合集
这里写目录标题 209.长度最小的子数组题目思路代码 3. 无重复字符的最长子串(medium)题目思路 11. 最大连续 1 的个数 III题目思路 1658. 将 x 减到 0 的最⼩操作数题目思路代码 904. 水果成篮题目思路代码 438.找到字符串中所有字母的异位词题目思路代码…...

【C++】STL详解(九)—— set、map、multiset、multimap的介绍及使用
📝个人主页:Sherry的成长之路 🏠学习社区:Sherry的成长之路(个人社区) 📖专栏链接:C学习 🎯长路漫漫浩浩,万事皆有期待 上一篇博客:【C】STL…...

计组—— I/O系统
📕:参考王道课件 目录 一、I/O系统的基本概念 1.什么是“I/O”? 编辑2.主机如何和I/O设备进行交互? 3.I/O控制方式 (1)程序查询方式 (2)程序中断方式 (3&#x…...

基于vc6+sdk51开发简易文字识别转语音的程序
系统:window7 软件:vc6.0 目的:简易文字转语音真人发声 利用2023国庆小长假,研究如何将文言转语音,之前在网上查询相关知识,大致了解微信语音转换,翻译官之类软件的原理,但要加入神…...
DevOps:自动化部署和持续集成/持续交付(CI/CD)
DevOps:自动化部署和持续集成/持续交付(CI/CD) 在现代软件开发领域,DevOps(Development和Operations的组合)已经成为一个不可或缺的概念。它代表了一种将软件开发和运维(Operations)…...

专业图标制作软件 Image2icon 最新中文 for mac
Image2Icon是一款用于Mac操作系统的图标转换工具。它允许用户将常见的图像文件(如PNG、JPEG、GIF等)转换为图标文件(.ico格式),以便在Mac上用作应用程序、文件夹或驱动器的自定义图标。 以下是Image2Icon的一些主要功…...
数据结构:顺序表
SeqList.h #pragma once #include<stdio.h> #include<assert.h> #include<stdlib.h>typedef int SLDataType; //#define NULL 0typedef struct SeqList {SLDataType* a;int size;//顺序表中存储的有效元素的个数int capacity;//空间的大小 }SL;void SLInit(…...

僵尸进程的产生与处理
僵尸进程(Zombie Process)是指在操作系统中已经完成了执行,但其父进程尚未调用wait()或waitpid()来获取其终止状态的子进程。当一个进程结束时,操作系统会保留该进程的一些基本信息,包括进程ID(PID…...
TouchEffects - Android View点击特效
官网 GitHub - likaiyuan559/TouchEffects: Android View点击特效TouchEffects,几行代码为所有控件添加点击效果 项目简介 Android View点击特效TouchEffects,几行代码为所有控件添加点击效果 TouchEffects能够帮助你更快速方便的增加点击时候的效果,TouchEffect…...
从ContinuousEventTimeTrigger/ContinuousProcessingTimeTrigger代码看如何实现一个自定义的触发器
背景 当我们想要实现提前触发计算的触发器时,我们可以使用ContinuousEventTimeTrigger/ContinuousProcessingTimeTrigger作为触发器达到比如几分钟触发一次计算并发送计算结果的类,我们本文就从代码角度解析下实现自定义触发器的一些注意事项 Continuo…...

Linux 5种网络模型
[参考]:《黑马程序员Redis》https://www.bilibili.com/video/BV1cr4y1671t/?p166&share_sourcecopy_web&vd_source9e65300ccca322aeb367bb1eb677b0fc [参考]:《操作系统》 [参考]:《UNIX网络编程》 为了避免用户应用导致冲突甚至内…...

10.1 调试事件读取寄存器
当读者需要获取到特定进程内的寄存器信息时,则需要在上述代码中进行完善,首先需要编写CREATE_PROCESS_DEBUG_EVENT事件,程序被首次加载进入内存时会被触发此事件,在该事件内首先我们通过lpStartAddress属性获取到当前程序的入口地…...

Linux系统常用指令篇---(一)
Linux系统常用指令篇—(一) 1.cd指令 Linux系统中,磁盘上的文件和目录被组成一棵目录树,每个节点都是目录或文件。 语法:cd 目录名 功能:改变工作目录。将当前工作目录改变到指定的目录下。 (简单理解为进入指定目录下) 举例: cd .. : 返…...

【初识Linux】:常见指令(1)
朋友们、伙计们,我们又见面了,本期来给大家解读一下有关Linux的基础知识点,如果看完之后对你有一定的启发,那么请留下你的三连,祝大家心想事成! C 语 言 专 栏:C语言:从入门到精通 数…...

STM32复习笔记(四):看门狗
目录 (一)简介 (二)IWDG IWDG的CUBEMX工程配置 IWDG相关函数(非常少,所以直接贴上来): (三)WWDG (一)简介 看门狗分为独立看门…...

51c自动驾驶~合集58
我自己的原文哦~ https://blog.51cto.com/whaosoft/13967107 #CCA-Attention 全局池化局部保留,CCA-Attention为LLM长文本建模带来突破性进展 琶洲实验室、华南理工大学联合推出关键上下文感知注意力机制(CCA-Attention),…...

定时器任务——若依源码分析
分析util包下面的工具类schedule utils: ScheduleUtils 是若依中用于与 Quartz 框架交互的工具类,封装了定时任务的 创建、更新、暂停、删除等核心逻辑。 createScheduleJob createScheduleJob 用于将任务注册到 Quartz,先构建任务的 JobD…...

相机从app启动流程
一、流程框架图 二、具体流程分析 1、得到cameralist和对应的静态信息 目录如下: 重点代码分析: 启动相机前,先要通过getCameraIdList获取camera的个数以及id,然后可以通过getCameraCharacteristics获取对应id camera的capabilities(静态信息)进行一些openCamera前的…...
【HarmonyOS 5 开发速记】如何获取用户信息(头像/昵称/手机号)
1.获取 authorizationCode: 2.利用 authorizationCode 获取 accessToken:文档中心 3.获取手机:文档中心 4.获取昵称头像:文档中心 首先创建 request 若要获取手机号,scope必填 phone,permissions 必填 …...
在web-view 加载的本地及远程HTML中调用uniapp的API及网页和vue页面是如何通讯的?
uni-app 中 Web-view 与 Vue 页面的通讯机制详解 一、Web-view 简介 Web-view 是 uni-app 提供的一个重要组件,用于在原生应用中加载 HTML 页面: 支持加载本地 HTML 文件支持加载远程 HTML 页面实现 Web 与原生的双向通讯可用于嵌入第三方网页或 H5 应…...

七、数据库的完整性
七、数据库的完整性 主要内容 7.1 数据库的完整性概述 7.2 实体完整性 7.3 参照完整性 7.4 用户定义的完整性 7.5 触发器 7.6 SQL Server中数据库完整性的实现 7.7 小结 7.1 数据库的完整性概述 数据库完整性的含义 正确性 指数据的合法性 有效性 指数据是否属于所定…...

STM32HAL库USART源代码解析及应用
STM32HAL库USART源代码解析 前言STM32CubeIDE配置串口USART和UART的选择使用模式参数设置GPIO配置DMA配置中断配置硬件流控制使能生成代码解析和使用方法串口初始化__UART_HandleTypeDef结构体浅析HAL库代码实际使用方法使用轮询方式发送使用轮询方式接收使用中断方式发送使用中…...

day36-多路IO复用
一、基本概念 (服务器多客户端模型) 定义:单线程或单进程同时监测若干个文件描述符是否可以执行IO操作的能力 作用:应用程序通常需要处理来自多条事件流中的事件,比如我现在用的电脑,需要同时处理键盘鼠标…...
【安全篇】金刚不坏之身:整合 Spring Security + JWT 实现无状态认证与授权
摘要 本文是《Spring Boot 实战派》系列的第四篇。我们将直面所有 Web 应用都无法回避的核心问题:安全。文章将详细阐述认证(Authentication) 与授权(Authorization的核心概念,对比传统 Session-Cookie 与现代 JWT(JS…...

HTTPS证书一年多少钱?
HTTPS证书作为保障网站数据传输安全的重要工具,成为众多网站运营者的必备选择。然而,面对市场上种类繁多的HTTPS证书,其一年费用究竟是多少,又受哪些因素影响呢? 首先,HTTPS证书通常在PinTrust这样的专业平…...