Java反射机制,动态代理,hook以及在Retrofit源码中的应用

news/2024/5/19 5:19:46 标签: java, retrofit, 反射, 动态代理, hook

1.反射的基础知识:
Java的反射机制是指在程序的运行状态中,可以构造任意一个类的对象,可以了解任意一个对象所属的类,可以了解任意一个类的成员变量和方法,可以调用任意一个对象的属性和方法。这种动态获取程序信息以及动态调用对象的功能称为Java语言的反射机制。
反射的基本操作:
获取类对象

获取一个类的 Class 对象有多种方法:

1.1.1使用 Class.forName(String className) 方法,需要传入类的全限定名,会返回该类的 Class 对象。

1.1.2使用类名.class,例如 String.class。

1.1.3使用对象.getClass() 方法,返回对象所属的类的 Class 对象。
获取类的属性

1.2使用 Class 类的 getField(String name) 方法可以获取类的指定公有属性;使用 getDeclaredField(String name) 方法可以获取类的指定属性(无论是否公有)。

1.3获取类的方法

使用 Class 类的 getMethod(String name, Class… parameterTypes) 方法可以获取类的指定公有方法;使用 getDeclaredMethod(String name, Class… parameterTypes) 方法可以获取类的指定方法(无论是否公有)。

1.4获取类的构造函数

使用 Class 类的 getConstructor(Class… parameterTypes) 方法可以获取类的指定公有构造函数;使用 getDeclaredConstructor(Class… parameterTypes) 方法可以获取类的指定构造函数(无论是否公有)。

1.5动态调用方法

通过反射可以动态调用类的方法,例如:

Class clazz = Class.forName(“com.example.MyClass”);
Object obj = clazz.newInstance();
Method method = clazz.getMethod(“myMethod”, String.class);
method.invoke(obj, “parameter”);
上述代码中,我们先获取 MyClass 的 Class 对象,然后创建 MyClass 的一个实例,接着获取 myMethod 方法的 Method 对象,最后使用 invoke 方法调用 myMethod 方法并传入参数。
2.Retrofit源码
在这里插入图片描述
讲上图之前你要本身会使用Retrofit,我们知道Retrofit 的接口是统一集成在一个接口文件中的,所以上图中的service参数就是
类似下图的接口类
在这里插入图片描述
动态代理主要就是利用Proxy.newProxyInstance方法生成代理类,其必须实现InvocationHandler invoke方法,逻辑是如果接口中的方法,记住是所有方法,这也是和使用接口继承去实现的区别,接口中定义的方法统一操作,是动态代理 接口类的一大特征,如果方法是object的方法就直接invoke传入当前对象和参数,如果不是接着往下走,判断当前平台是java,android,还是linux的,如果是系统默认的方法就invoke系统默认的方法,否则用loadServiceMethod来invoke,然后我们看一下下图的loadServiceMethod方法,最后调用了ServiceMethod.parseAnnotations
在这里插入图片描述
然后是HttpServiceMethod.parseAnnotations
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
经过上图的几次跳转后,下图红色箭头的就是最后的CallAdapter接口承接的对象即来自于 callAdapterFactories
在这里插入图片描述
由下图我们找到了是platform.createDefaultCallAdapterFactories产生的
在这里插入图片描述
那我们平台是android 就是android24了,CompletableFutureCallAdapterFactory
和 DefaultCallAdapterFactory
在这里插入图片描述
然后就是代理的实质代码了call.enqueue
在这里插入图片描述
最后会在build里面execute,完成整个流程,代理执行完毕。
在这里插入图片描述
提一句,从下图可以看出来Proxy.newProxyInstance动态代理里面的实现用到了反射
在这里插入图片描述
3.关于Hook的我做了一个demo,英文翻译过来是钩子,用一个成语表达就是偷梁换柱。用kotlin实现的,比起java会有点不一样的:
看下图,先说要干什么,hook 一下 view的setOnClicklistener的View.OnclickListener。那怎么做呢,因为OnclickListener是接口,可以用动态代理生成,比继承重写接口方便一些。可能你还会有疑问那我直接用一个View.OnclickListener继承类来set进行不就行了,我要说的是这个测试用例只是为了方便描述,这是这个接口有set ,对外是开放的,你可以按你想的那样弄,假如它是不对外开放的呢,hook是不是就有意义了,好了,开始做吧。

在这里插入图片描述
首先我们先看一下view.setOnClickListener方法里面的情况。下图中我们要hook的对象被getListenerInfo()所持有,
在这里插入图片描述
那就看看getListenerInfo()返回的对象是
在这里插入图片描述
是view的一个属性mListenerInfo
在这里插入图片描述
ListenerInfo是View的一个静态内部类,而它的属性mOnClickListener正是我们要找的hook对象了。
在这里插入图片描述
那思路就是利用反射来拿到mListenerInfo对象,拿到它的属性mOnClickListener并替换成 我们动态代理生成的接口。需要注意的是args,kotlin是需要加的。查了查好像说是数组展开,没理解,知道的可以评论区评论。

在这里插入图片描述
然后我们回过头来看反射机制是 基础,动态代理里面也是用反射来实现的,动态代理对类里面的所有方法统一动态加入操作代码,java使用更多是对接口的,是生成了你想往里面塞的代码,hookhook对象勾出来,换成自己想塞进去的,是一种手法。


http://www.niftyadmin.cn/n/4932594.html

相关文章

环境与分支的详细介绍及其关联(开发、测试、预发布、生产)

文章目录 前言一、开发环境(dev)二、测试环境(test)三、预发布环境(pre)四、生产环境(pro)五、环境与分支的关系总结 前言 在现代软件开发中,前端项目的开发和部署往往需…

Vc - Qt - 绘制窗口背景色

要在Qt中绘制一个背景颜色,你可以使用Qt的绘图功能来完成。下面是一种简单的方法: 步骤1:在你想要绘制背景颜色的QWidget(例如QMainWindow或QDialog)的派生类中,重写 它的paintEvent函数。步骤2&#xff1a…

Chisel 语言 - 小记

文章目录 Chisel 一种硬件描述语言,类似 verilog 本质是 Scala编程语言的一个包,类似于 numpy 是 Python 的一个包。 官网 : https://www.chisel-lang.orggithub: https://github.com/chipsalliance/chisel 同名的还有个 Facebook…

LeetCode150道面试经典题--判断子序列(简单)

1.题目 给定字符串 s 和 t ,判断 s 是否为 t 的子序列。 字符串的一个子序列是原始字符串删除一些(也可以不删除)字符而不改变剩余字符相对位置形成的新字符串。(例如,"ace"是"abcde"的一个子序…

安装paddlehub后测试运行报错

昨晚按照官网提示安装paddle和paddlehub后,运行paddle测试成功了。但是当我测试paddlehub的时候总是提示缺少模块,非常纳闷,后面发现是官方的坑,所以在这里做个笔记。 测试paddlehub报错: #安装paddlehub命令 pip in…

Golang字符串处理深入解析:探索 strings 标准库的全部方法

Golang 的 strings 标准库提供了许多用于处理字符串的函数。以下是一些主要的方法: Contains(s, substr string) bool: 检查字符串是否包含子串。ContainsAny(s, chars string) bool: 检查字符串是否包含字符集中的任何字符。ContainsRune(s string, r rune) bool:…

Android JNI2--C++基础

1,基础结构 C标准支持 #include <iostream>C语言的标准支持 #include <stdio.h> 命名空间 C的特性 std C系统的命名空间 也可以自定义 using namespace std; C中命名空间的作用类似于操作系统中的目录和文件的关系&#xff0c;由于文件很多&#xff0c;不便管…

OLTP和OLAP的区别以及使用场景

1、什么是OLTP&#xff1f; 全称OnLine Transaction Processing&#xff0c;联机事务处理系统&#xff0c;就是对数据的增删改查等操作 存储的是业务数据&#xff0c;来记录某类业务事件的发生&#xff0c;比如下单、支付、注册等 典型代表有Mysql、Oracle等数据库&#xff…