博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
iOS开发 - iOS的hook方案
阅读量:4107 次
发布时间:2019-05-25

本文共 1051 字,大约阅读时间需要 3 分钟。

在没有一个类的实现源码的情况下,我们如果想改变其中一个方法的实现,有哪些方法可以实现呢?

一、继承重写

这种方式,并不是严格意义上的hook,但是也可以达到改变方法实现的目的。但它在使用时,需要通过继承类这个对象来操作,限制比较大。对于不知道你的继承类存在的调用方来说,就毫无意义。

二、借助类别重名方法

这种方式的话,是覆盖原有的实现,只会保留重写的功能。因为原有的功能

三、Method Swizzling

Method Swizzling 原理

在Objective-C中调用一个方法,其实是向一个对象发送消息,查找消息的唯一依据是selector的名字。利用Objective-C的动态特性,可以实现在运行时偷换selector对应的方法实现,达到给方法挂钩的目的。
每个类都有一个方法列表,存放着selector的名字和方法实现的映射关系。IMP有点类似函数指针,指向具体的Method实现。

我们可以利用 method_exchangeImplementations 来交换2个方法中的IMP,

我们可以利用 class_replaceMethod 来修改类,

我们可以利用 method_setImplementation 来直接设置某个方法的IMP,

……

归根结底,都是偷换了selector的IMP,如下图所示:

Method Swizzling 实践



举个例子好了,我想钩一下NSArray的lastObject 方法,只需两个步骤。

第一步:给NSArray加一个我自己的lastObject

乍一看,这不递归了么?原因后面阐述。

第二步:调换IMP
控制台输出Log:
现在我们来分析上面提到的递归的问题。
由于交换了方法的实现  @selector(lastObject) 对应 IMP(myLastObject), @selector(myLastObject) 对应 IMP(lastObject)
所以执行 NSString *string [array lastObject]; 时,实际是去调用myLastObject, 函数myLastObject里的代码id ret [self myLastObject]; 将会执行真的 [self lastObject] 。
这里就多亏了OC的动态机制以及消息特性
如果不执行IMP的交换操作,调用
[array myLastObject]; 就会陷入死循环。
 四、delegate的hook

四、fishhook

你可能感兴趣的文章
【Python】学习笔记——-7.3、继承和多态
查看>>
【Python】学习笔记——-7.4、获取对象信息
查看>>
【Python】学习笔记——-7.5、实例属性和类属性
查看>>
Linux设备模型(总线、设备、驱动程序和类)之四:class_register
查看>>
git中文安装教程
查看>>
虚拟机 CentOS7/RedHat7/OracleLinux7 配置静态IP地址 Ping 物理机和互联网
查看>>
弱类型、强类型、动态类型、静态类型语言的区别是什么?
查看>>
Struts2技术内幕图书 转载
查看>>
Java异常分类
查看>>
项目中的jackson与json-lib使用比较
查看>>
Jackson Tree Model Example
查看>>
j2ee-验证码
查看>>
日志框架logj的使用
查看>>
js-高德地图规划路线
查看>>
常用js收集
查看>>
mydata97的日期控件
查看>>
如何防止sql注入
查看>>
maven多工程构建与打包
查看>>
springmvc传值
查看>>
Java 集合学习一 HashSet
查看>>