news 2026/5/10 14:02:49

Dexmaker实战指南:Android动态代码生成的终极武器

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
Dexmaker实战指南:Android动态代码生成的终极武器

Dexmaker是一个专为Android平台设计的Java语言API,用于在编译时或运行时生成针对Dalvik虚拟机的代码。与cglib或ASM等传统代码生成库不同,Dexmaker直接创建Dalvik的.dex文件而非Java的.class文件,这在Android开发领域具有独特价值。

【免费下载链接】dexmaker项目地址: https://gitcode.com/gh_mirrors/dex/dexmaker

核心功能深度解析

动态字节码生成能力

Dexmaker提供了接近底层的API,这些API镜像了Dalvik字节码规范,让你能够精确控制生成的字节码。代码是逐条指令生成的,如果你需要抽象语法树,需要自行构建。由于它使用Dalvik的dx工具作为后端,你可以免费获得高效的寄存器分配和常规/宽指令选择。

应用场景示例

  • 在运行时生成适配特定业务逻辑的类
  • 为测试框架创建动态Mock对象
  • 实现AOP(面向切面编程)功能

Mockito集成支持

Dexmaker通过生成Dalvik字节码类代理,让你能够在Android项目中使用Mockito模拟库。只需在androidTestImplementation中添加对dexmaker-mockito的依赖,就可以在Android Instrumentation测试中使用Mockito。

高级Mocking功能

从Android "P"开始,使用dexmaker-mockito-inline库可以模拟final类和final方法。如果你在运行Android P或更高版本的设备或模拟器上执行测试,可以添加对dexmaker-mockito-inline的依赖,然后就可以在Android Instrumentation测试中使用正常的Mockito API来模拟final类和final方法。

实际应用案例演示

基础代码生成示例

让我们通过一个简单的HelloWorld示例来了解Dexmaker的基本用法:

public final class HelloWorldMaker { public static void main(String[] args) throws Exception { DexMaker dexMaker = new DexMaker(); // 生成HelloWorld类 TypeId<?> helloWorld = TypeId.get("LHelloWorld;"); dexMaker.declare(helloWorld, "HelloWorld.generated", Modifier.PUBLIC, TypeId.OBJECT); generateHelloMethod(dexMaker, helloWorld); // 创建dex文件并加载 File outputDir = new File("."); ClassLoader loader = dexMaker.generateAndLoad(HelloWorldMaker.class.getClassLoader(), outputDir, outputDir); Class<?> helloWorldClass = loader.loadClass("HelloWorld"); // 在进程中执行新生成的代码 helloWorldClass.getMethod("hello").invoke(null); } private static void generateHelloMethod(DexMaker dexMaker, TypeId<?> declaringType) { // 查找过程中需要的类型 TypeId<System> systemType = TypeId.get(System.class); TypeId<PrintStream> printStreamType = TypeId.get(PrintStream.class); // 在declaringType上识别'hello()'方法 MethodId hello = declaringType.getMethod(TypeId.VOID, "hello"); // 在dexMaker上声明该方法 Code code = dexMaker.declare(hello, Modifier.STATIC | Modifier.PUBLIC); // 预先声明所有需要的局部变量 Local<Integer> a = code.newLocal(TypeId.INT); Local<Integer> b = code.newLocal(TypeId.INT); Local<Integer> c = code.newLocal(TypeId.INT); Local<String> s = code.newLocal(TypeId.STRING); Local<PrintStream> localSystemOut = code.newLocal(printStreamType); // 生成具体的字节码指令 code.loadConstant(a, 0xabcd); code.loadConstant(b, 0xaaaa); code.op(BinaryOp.SUBTRACT, c, a, b); MethodId<Integer, String> toHexString = TypeId.get(Integer.class).getMethod(TypeId.STRING, "toHexString", TypeId.INT); code.invokeStatic(toHexString, s, c); FieldId<System, PrintStream> systemOutField = systemType.getField(printStreamType, "out"); code.sget(systemOutField, localSystemOut); MethodId<PrintStream, Void> printlnMethod = printStreamType.getMethod( TypeId.VOID, "println", TypeId.STRING); code.invokeVirtual(printlnMethod, null, localSystemOut, s); code.returnVoid(); } }

斐波那契数列生成器

为了进一步说明API的使用方式,让我们用Dexmaker生成一个等效于以下Java源码的类:

public class Fibonacci { public static int fib(int i) { if (i < 2) { return i; } return fib(i - 1) + fib(i - 2); } }

项目集成与配置

依赖管理

对于Mockito支持,通过Maven下载最新的jar包:

<dependency> <groupId>com.linkedin.dexmaker</groupId> <artifactId>dexmaker-mockito</artifactId> <version>2.28.3</version> <type>pom</type> </dependency>

或者使用Gradle:

androidTestImplementation 'com.linkedin.dexmaker:dexmaker-mockito:2.28.3'

快照版本使用

你可以使用快照构建来测试最新的未发布更改。只需将Sonatype快照仓库添加到你的Gradle脚本中:

repositories { maven { url "https://oss.sonatype.org/content/repositories/snapshots/" } }

进阶使用技巧

共享类加载器配置

如果一个类想要调用另一个类的包私有方法,它们需要共享一个类加载器。一个常见的情况是模拟类想要模拟原始类的包私有方法。使用setSharedClassLoader方法可以设置共享的类加载器。

受信任标记

在某些情况下,你可能需要将生成的类标记为受信任的,以便它们可以调用隐藏的API。这对于在系统类上进行特殊操作是必需的,因为这些类的真实方法可能会调用受限API。

常见问题与解决方案

兼容性问题

  • Android版本限制dexmaker-mockito-inline功能需要Android P引入的OS API,无法在旧版本Android上工作。

性能优化建议

  • 在生成大量类时,合理使用共享类加载器可以减少内存占用
  • 对于频繁使用的生成类,考虑预编译并缓存结果

最佳实践总结

Dexmaker作为Android平台上的动态代码生成工具,在测试框架开发、运行时代码生成等场景中发挥着重要作用。通过掌握其核心API和使用技巧,开发者可以构建更加灵活和强大的Android应用。记住,合理使用动态代码生成能够显著提升应用的适应性和可测试性。

【免费下载链接】dexmaker项目地址: https://gitcode.com/gh_mirrors/dex/dexmaker

创作声明:本文部分内容由AI辅助生成(AIGC),仅供参考

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/5/7 12:29:20

终极指南:llm-client TypeScript AI应用快速上手与实用技巧

终极指南&#xff1a;llm-client TypeScript AI应用快速上手与实用技巧 【免费下载链接】llm-client LLMClient - A Caching and Debugging Proxy Server for LLM Users and A Multi-LLM Client Library 项目地址: https://gitcode.com/gh_mirrors/ll/llm-client 构建AI…

作者头像 李华
网站建设 2026/5/3 5:37:57

Blender材质资源宝典:从零打造专业级3D作品

Blender材质资源宝典&#xff1a;从零打造专业级3D作品 【免费下载链接】awesome-blender &#x1fa90; A curated list of awesome Blender addons, tools, tutorials; and 3D resources for everyone. 项目地址: https://gitcode.com/GitHub_Trending/aw/awesome-blender …

作者头像 李华
网站建设 2026/5/1 11:39:56

智能送药小车数字字模实战指南:解决嵌入式显示核心难题

智能送药小车数字字模实战指南&#xff1a;解决嵌入式显示核心难题 【免费下载链接】智能送药小车F题数字字模资源说明分享 本仓库提供的是2021年全国大学生电子设计竞赛F题相关的技术资料——《智能送药小车&#xff08;F题&#xff09;数字字模.pdf》。这份文档专为参赛团队设…

作者头像 李华
网站建设 2026/5/1 6:20:50

Polymaps地图库终极指南:3步创建动态交互地图

Polymaps是一个完全免费的JavaScript地图库&#xff0c;专门用于在现代浏览器中制作动态交互地图。这个开源地图开发教程将带你快速掌握Polymaps的核心功能和使用方法。 【免费下载链接】polymaps Polymaps is a free JavaScript library for making dynamic, interactive maps…

作者头像 李华
网站建设 2026/5/1 15:23:42

多租户工作流系统架构设计与性能优化终极指南

多租户工作流系统架构设计与性能优化终极指南 【免费下载链接】ruoyi-vue-pro &#x1f525; 官方推荐 &#x1f525; RuoYi-Vue 全新 Pro 版本&#xff0c;优化重构所有功能。基于 Spring Boot MyBatis Plus Vue & Element 实现的后台管理系统 微信小程序&#xff0c;支…

作者头像 李华
网站建设 2026/5/1 15:27:15

GBase 8a_SENSITIVE_DIRECTORY_ACCESS_DISABLE 参数简介

参数名&#xff1a;_sensitive_directory_access_disable参数 参数值&#xff1a;字符串类型&#xff0c;支持正则、分隔符号&#xff0c;默认ssh 参数可见范围&#xff1a;不可被show 功能说明&#xff1a; 安全漏洞审查中认为导出数据到/home/xxx/.ssh/目录中可能存在安全隐患…

作者头像 李华