Xposed加载JNI库

前言

不用介绍啦,何老师出手,黑科技降临。

在项目开发中,时常会用到 JNI 库,以提供一些特定的功能,而在 xposed 开发中,也会有这样的需求,然而,在 xposed 的条件下,要加载一个 so 可不是一件容易的事。

首先的问题是跨进程,由于 xposed 程序在执行时,xposed 模块与主包并不在同一进程,因此无法直接使用以下代码对 JNI 库进行加载:

如果这么做,那么只会得到一个 UnsatisfiedLinkError,因为在 xposed 进程所可以访问的空间内,找不到这个 so。

那么是否可以使用另一个加载方法,即 System.load 呢?如下的代码:

这样的代码在部分手机上可以工作,但是在部分手机上依然得到了一个 UnsatisfiedLinkError,但是具体的出错信息变了:

看出错信息,是在一个 64 位的进程内,加载了 32 位的库,于是在这里就会有一个时机的问题,需要先判断进程的位数,而在 64 位的设备上,默认的 xposed 进程也是 64 位的。以下就有两个解决方案:

方案一:编译 arm64-v8a 架构的库,然后在加载时,加载 64 位的库

这样就可以适应 64 位的设备。具体的位数判断可以反射 dalvik.system. VMRuntime 类,并且调用其中的 is64Bit 方法。

方案二:改变 JNI 库的加载时机,将初始化时的加载修改到 hook 到指定 32 位包的加载时进行加载。

由于自己的程序只有 32 位的 JNI 库,因此会加载为 32 位的应用,在自身被加载时加载 JNI 库,就可以顺利加载到 32 位的库了,此时即使设备是 64 位的,也可以正常加载到 32 位的库。

在加载完成后,再做一个简单的函数调用,或是实现 JNI_OnLoad 即可进行测试,最终实现的效果如下:

本文中所使用的代码工程已上传到 github(点击原文链接,一键传送),若有不清楚的地方,还是直接看代码吧。

本文分享自微信公众号 - Android群英传(android_heroes),作者:何晓杰

原文出处及转载信息见文内详细说明,如有侵权,请联系 yunjia_community@tencent.com 删除。

原始发表时间:2017-02-14

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • 一触即发——App启动优化最佳实践

    用户1907613
  • Bitmap.recycle引发的血案

    用户1907613
  • Android中的睡与不睡

    用户1907613
  • 读《深入理解Java虚拟机》解决实际问题及总结JDK和JVM整体架构

    以前看别人博客说看完《深入理解Java虚拟机》这本书并没有让自己的编程水平提高多少,不过却大大提高了自己的装逼水平。其实,我倒不这么认为,至少在我看完一遍这本书...

    Java小朔哥
  • 类加载器与双亲委派模型1 类加载器 2 双亲委派模型

    类加载器(ClassLoader)是Java语言的一项创新,也是Java流行的一个重要原因。 在类加载的第一阶段“加载”过程中,需要通过一个类的全限定名来获取...

    JavaEdge
  • 深入理解JVM(③)虚拟机的类加载器(双亲委派模型)

    先解释一下什么是类加载器,通过一个类的全限定名来获取描述该类的二进制字节流,在虚拟机中实现这个动作的代码被称为“类加载器(Class Loader)”。

    纪莫
  • 持续3分钟 - Java -09

    JVM 类加载机制分为五个部分:加载,验证,准备,解析,初始化,下面我们就分别来看一下这五个过程。

    子乾建建-Jeff
  • Java底层-类加载子系统(二)

    其中初始化是类加载的最后一步,使用和卸载不属于类加载的过程、此外在这些动作中, 加载阶段是唯一一个用户可以通过类加载器参与的阶段(非数组类的加载), 后续的其余...

    每天学Java
  • 带你搞懂双亲委派机制

    之前详细介绍了Java类的整个加载过程(类加载机制你真的了解吗?)。虽然,篇幅较长,但是也不要被内容吓到了,其实每个阶段都可以用一句话来概括。

    烟雨星空
  • 看完这篇JVM类加载器,我也能吊打面试官了

    第二、三层类加载器为Java语言实现,用户也可以自定义类加载器 查看本地类加载器的方式如下:

    JavaEdge

扫码关注云+社区

领取腾讯云代金券