android开发日常:使用jni执行任何二进制文件
什么是 JNI?
JNI,即 Java Native Interface 的缩写,通过使用 Java 本地接口编写程序,可以确保代码在不同平台上方便移植。从 Java 1.1 开始,JNI 标准成为 Java 平台的一部分,它允许 Java 代码与其他语言编写的代码进行交互。JNI 最初是为了本地已编译的语言,尤其是 C 和 C++ 设计的,但它并不妨碍你使用其他编程语言,只要调用约定受支持即可。使用 Java 与本地已编译的代码交互,通常会丧失平台的可移植性。然而,在某些情况下,这样做是可以接受的,甚至是必要的。例如,使用一些旧的库,与硬件或操作系统进行交互,或者为了提高程序的性能。JNI 标准至少保证本地代码能在任何 Java 虚拟机环境中工作。
在哪里见过?native 关键字。一个 native 方法就是一个 Java 调用非 Java 代码的接口。一个 native 方法是指该方法的实现由非 Java 语言实现,比如用 C 或 C++ 实现。在定义一个 native 方法时,不提供实现体(类似于定义一个 Java Interface),因为其实现体是由非 Java 语言在外部实现的。这主要是因为 Java 无法对操作系统底层进行操作,但可以通过 JNI(Java Native Interface)调用其他语言来实现底层的访问。
提出问题
很多时候使用 Kotlin 或 Java 开发 Android 时都离不开访问 /data/data/com.xxx.xxx/ 下的文件,由于 Linux 的不可控因素,在高版本 Android 系统中 Runtime.exec("su") 已经失效。那么该如何使用 root 权限去执行应用包下的二进制文件呢?
一些前提条件
使用 native 是少不了 NDK 包的,通过 Preferences(Settings) > Appearence & Behavior > System Settings > Android SDK 中的 SDK Tools 下载 NDK 与 CMake,具体如下图:

解决方案架构
在创建项目时使用 native C++ 模板进行创建;在 /src/main/ 包下会出现 cpp 与 java 两种语言的核心包;进入 /src/main/cpp/native-lib.cpp 中,可以看到系统已自动生成了一个 C++ 函数;
System Fork
现在使用我们二年级学过的 C++ 知识来写一个 Linux 操作,让 system() 函数去执行:
#include #include #include #include #includevoid Shell(){char shell[64];sprintf(shell,"sh /data/data/com.example.jni/start.sh");system(shell);return nullptr;}
这样我们就创建了一个 Shell() 方法。
JNI调用
宣小二
宣小二:媒体发稿平台,自媒体发稿平台,短视频矩阵发布平台,基于AI驱动的企业自助式投放平台。
21 查看详情
将 Shell() 方法挂载到 JNI 实例中:
extern "C"JNIEXPORT void JNICALLJava_com_example_jni_utils_shellUtil_execShell(JNIEnv env, jobject thiz) {Shell();}
回到需要调用的工具类(utils.shellUtil)中,写入调用:
static {System.loadLibrary("native_lib");}private static native void shell(); //使用native关键字调取
这里的 System.loadLibrary("native_lib") 的意思为:调用你 build 之后生成的 libnative_lib.so SO库。
so库在哪里
编写完 C++ native lib 之后进行 build 操作可以在文件目录 /build/intermediates/merged_native_libs/debug/out/lib 下找到对应不同操作系统的 so 库文件。将它们复制到你的 libs(与 src 同级目录) 下后再 run 你的项目即可完成调用。

多线程
至此,已经完成了 native 库的编写与运行,你应该对 JNI 也有了一定的了解。但很多情况下我们不希望 被运行的二进制文件 阻碍 安卓主线程,这时候,需要使用到多线程对二进制文件的运行进行处理。我们可以在 native-lib.cpp 中这样处理:
#includeinclude
include
include
include
include
void* Shell(){char shell[64];//fork拷贝sprintf(shell,"sh /data/data/com.example.jni/start.sh");system(shell);return nullptr;}
extern "C"JNIEXPORT void JNICALLJava_com_example_jni_utils_shellUtil_execShell(JNIEnv env, jobject thiz) {//创建线程idpthread_t tid;//启动线程pthread_create(&tid,nullptr,reinterpret_cast<void>(Shell),nullptr);}
通过 pthread 函数库进行线程处理,这样就保障了安卓应用主线程的线程安全,与并行的效率。
如何停止线程?管道通信
我们在小学三年级的 Linux操作系统 课程中已经知道了 system() 命令的执行过程是 fork子进程 执行二进制,这样就带来一个问题:我的二进制文件需要指定一个配置来启动的话就读取不到被设定为 read only 的文件夹内的资源。
如何解决?我也不会,希望有大佬能指点江山。
以上就是Android开发日常:使用JNI执行任何二进制文件的详细内容,更多请关注创想鸟其它相关文章!
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/468917.html
微信扫一扫
支付宝扫一扫