当前位置: 首页 > news >正文

Frida使用java.lang.reflect.Array类打印Java反射数组

前言

通过frida进行java反射调用String的getBytes方法,得到了[B类型的引用
无论如何尝试都无法打印输出其内容,甚至java.util.Arrays类的相关方法也无法正确输出

之后刷到这篇文章了解到java.lang.reflect.Array类用于反射打印数组
https://www.cnblogs.com/liwxmyself/p/14758632.html
进而了解到另一篇文章,详细介绍了相关方法
https://www.cnblogs.com/pony1223/p/7470382.html

总之,通过反射获取到的数组,不能直接通过Arrays类操作,应该通过reflect包下的Array类操作

案例

创建Java字符串后,有如下3个字节数组

  • result 直接通过反射调用getBytes方法获取字符串对应的字节数组
  • result2 调用java方法getBytes后转换为字节数组类型
  • result3 调用getBytes后不进行其他操作

分别通过Typeof打印类型以及直接打印结果

                //反射操作Stringvar str=Java.use("java.lang.String").$new("demo")var result=Reflection.invokeMethod("java.lang.String","getBytes",str, [], [])//[B byte[]数组,无法直接打印var result2=Java.cast(str.getBytes(),Java.use("[B"))//转换为byte[]数组,无法直接打印var result3=str.getBytes()//可直接打印//打印类型console.log("Typeof result:", typeof result);//Typeof result: objectconsole.log("Typeof result2:", typeof result2);//Typeof result2: objectconsole.log("Typeof result3:", typeof result3);//Type of result3: object//打印结果console.log("[+] result:", result);//[+] result: [B@7ab81e8console.log("[+] result2:", result2);//[+] result2: [B@70b3001console.log("[+] result3:", result3);//[+] result3: 100,101,109,111

打印结果如下,可以发现均为object类型,但是直接打印结果略有区别
只有result3可以直接显示内容,其他均为引用
在这里插入图片描述
之后利用java.lang.reflect.Array类的静态方法

  • Object get(Object[],int index) 反射获取任意类型数组指定下标的元素
  • byte getByte(Object[],int index) 反射获取byte[]类型数组指定下标的元素
                //反射调用得到的数组类型需要使用java.lang.reflect.Array类进行操作,而非java.util.Arrays类const reflectArray = Java.use("java.lang.reflect.Array");//基本操作console.log(typeof reflectArray.get(result,0))//objectconsole.log(typeof reflectArray.getByte(result,0))//numberconsole.log("length:",reflectArray.getLength(result));//length: 4//遍历for(var i=0;i<reflectArray.getLength(result);i++){console.log("result["+i+"]:",reflectArray.getByte(result,i));}for(var i=0;i<reflectArray.getLength(result2);i++){console.log("result2["+i+"]:",reflectArray.getByte(result2,i));}

打印结果如下
get获取的是object类型
getByte获取到的是number类型
并且成功遍历
在这里插入图片描述

其中invokeMethod封装如下

// 核心:通过 ClassLoader 获取类引用
function getClass(className, classLoader) {try {if (classLoader) return classLoader.loadClass(className);// 使用指定的 ClassLoaderelse return Java.use(className).class;// 默认使用全局 ClassLoader} catch (e) {console.error(`[Reflection] getClass error: ${e}`);return null;}
}/*** 调用实例方法(支持指定 ClassLoader)* @param {string} className      类名* @param {string} methodName     方法名* @param {Object} instance      实例对象* @param {Array<Class>} paramTypes  参数类型* @param {Array<any>} paramValues 参数值* @param {ClassLoader} [classLoader] 类加载器(可选)*/invokeMethod: function (className, methodName, instance, paramTypes, paramValues, classLoader) {try {const clazz = getClass(className, classLoader);if (!clazz) return null;const method = clazz.getDeclaredMethod(methodName,Java.array('java.lang.Class', paramTypes));method.setAccessible(true);const jArgs = Java.array('java.lang.Object', paramValues);return method.invoke(instance, jArgs);} catch (e) {console.error(`[Reflection] invokeMethod error: ${e}`);return null;}},

相关文章:

  • Mkdocs页面如何嵌入PDF
  • 【Prometheus】业务指标与基础指标的标签来源差异及设计解析(扩展版)
  • 【Prometheus】深入解析 Prometheus 特殊标签 `__param_<name>`:动态抓取参数的艺术
  • var、let、const的区别
  • WPF MVVM入门系列教程(六、ViewModel案例演示)
  • 华为设备链路聚合实验:网络工程实战指南
  • Notepad++中XML格式化插件介绍
  • GPT与LLaMA:两大语言模型架构的深度解析与对比
  • console-chat-gpt开源程序是用于 AI Chat API 的 Python CLI
  • Android学习总结之Binder篇
  • Linux 下MySql主从数据库的环境搭建
  • RDK X5 交叉编译OSS\QT\opencv\openssl
  • 【Rust模块管理】Rust包、crate与模块管理
  • 深入探讨C++日志模块设计与实现
  • 华为昇腾910B通过vllm部署InternVL3-8B教程
  • 01 一文了解大数据存储框架:数据库、数据仓库、数据集市、数据网格、数据湖、数据湖仓
  • 设计一个分布式系统:要求全局消息顺序,如何使用Kafka实现?
  • 【面板数据】公开整理-各省刑事案件统计数据集(2011-2023年)
  • 锁相环HMC830的调试
  • taro的学习记录
  • 重庆大学通报本科生发14篇SCI论文处理结果
  • 伤员回归新援融入,海港逆转海牛重回争冠集团
  • 中华人民共和国和俄罗斯联邦关于全球战略稳定的联合声明
  • 北约年度报告渲染所谓“中国核威胁”,国防部回应
  • 国家发改委副主任谈民营经济促进法:以法治的稳定性增强发展的确定性
  • 上海:5月8日起5年以上首套个人住房公积金贷款利率下调至2.6%