Swift 从获取所有 NSObject 对象聊起:ObjC、汇编语言以及底层方法调用链(一)

在这里插入图片描述

概览

Swift 语言给我们的印象是:简洁、现代化和可以“心安神泰”的完全信赖。不过,在一些特殊情况下我们唯有进入 Swift 底层的动态世界方能真正地“随遇而安”。

在这里插入图片描述

保安局“刘局长”曾语重心长的教导过我们:“非常时期,用非常方法!”。所以,这里就让我们彻底沉浸到 Swift 那深不见底的“千尺冰寒”中,来探寻 Objective-C 和汇编语言的奇妙世界吧!

在本篇博文中,您将学到以下内容:

  • 概览
  • 1. 获取当前所有 NSObject 对象的基本原理
  • 2. NSObject.init() 里面到底做了啥?
  • 3. 啥都不做也不行?你们是要闹哪样?
  • 总结

在下一篇博文中,我们将继续本次冒险之旅来进一步揭秘:为什么在 NSObject 构造器 init 方法上做钩子会如此“命运坎坷”,以及“逆天改命”的各种黑魔法!

好了,废话少叙!让我们马上开始奇妙的底层探险吧!

Let‘s Dive in!!!😉


1. 获取当前所有 NSObject 对象的基本原理

首先第一个问题是:为什么要获取 App 当前运行时所有的 NSObject 对象?

答案有两个:

  1. 好玩!
  2. 方便对它们做钩子(Hook)从而进一步实现自己的“天马行空”之梦想。

从 Xcode 的调试内存图(Debug Memory Graph)中,我们可以可以大致领略到 App 运行中那星罗棋布的海量对象实例:

在这里插入图片描述

那么第二个问题是:我们怎样才能像 Xcode 那样在运行时探查所有 NSObject 对象的实例呢?

一种方案是“古老”的 SWIZZ 方法,它存在于 Objective-C 语言的“远古”时代。

所谓 SWIZZ 是一种对系统的“欺骗”,一种运行时的“诡计”。它是利用 Swift 底层 ObjC 语言的动态特性实现在 App 进程活跃时修改 ObjC 运行时(Runtime)内容的方法。

在这里插入图片描述

SWIZZ 的一种常见应用就是钩子(Hook)。所谓钩子就是在 App 运行时动态的“勾住”(截获)对象方法从而起到监视、记录甚至修改对象原有方法的目的。

我们知道 NSObject 对象的诞生都必须经过 NSObject 类实例的构造器,也就是 NSObject.init() 方法来完成。那么如果我们在该方法上设置一个钩子,那么不就可以得偿所愿监控到所有 NSObject 对象了吗?

在这里插入图片描述

以下是我们的第一次尝试:

class func tryHookNSObjectInit() {let oldInitSel = #selector(NSObject.init)let oldInitMethod = class_getInstanceMethod(NSObject.self, oldInitSel)!let closure = {obj, sel inprint(obj)} as @convention(block) (NSObject, Selector) -> Voidlet closureObject = unsafeBitCast(closure, to: AnyObject.self)let closureImp = imp_implementationWithBlock(closureObject)method_setImplementation(oldInitMethod, closureImp)
}

如上所示,我们通过 SWIZZ 机制将 NSObject 构造器 init() 方法非常 easy 的替换成了自己的 closure 闭包。

不过,假若你胆敢运行上面的代码,你绝对会“死得很惨”:

在这里插入图片描述

因为你可能还不知道,你此时正“命犯天煞孤星”:多处“隐秘”陷阱正在一起合计着准备把你撕扯的体无完肤。

也许有些小伙伴们会认为崩溃的原因很“一目了然”:因为我们没有在钩子方法中调用原来的 NSObject 构造器。

真相果真如此吗?

为了解答这个问题,我们先要来看看在 NSObject 的构造器里到底做了些神马?

2. NSObject.init() 里面到底做了啥?

首先,清除之前的钩子代码。然后在 Xcode 中新建符号断点(Symbolic Breakpoint),姑且就中断在 NSObject.init() 方法的第一行吧:

在这里插入图片描述

运行 App,系统会在某个 NSObject 对象创建时立即中断下来。

在我测试的例子里,中断会发生在一个线程(NSThread)对象创建时:

在这里插入图片描述

进入 NSObject 构造器 init 方法,你会发现里面其实只有一条返回指令(ret)真的是“空空如也”!
在这里插入图片描述

所以说,在钩子方法中没有调用原来的 NSObject 构造器这并不算一个问题,因为确切的说默认的 NSObject 构造器方法里“空无一物”,调不调用几乎是无所谓的事。

3. 啥都不做也不行?你们是要闹哪样?

为了充分发挥小伙伴们那名侦探般“灵气逼人”的洞察力,我们现在改变策略:不打印对象本身,而是尝试打印对象的地址试试。

let closure = {obj, sel inlet address = Unmanaged.passRetained(obj).toOpaque()print(address)
} as @convention(block) (NSObject, Selector) -> Void

运行可以发现,我们在打印出第一个对象的地址后就很快又“GG”了:

在这里插入图片描述

看来打印对象地址这种“信手拈来”的简单操作也不行,那请允许我们退“一万步”:将钩子方法替换为一个“空”闭包总行了吧?

let closure = {obj, sel in// 空空如也!            
} as @convention(block) (Any, Selector) -> Void

不幸的是,结果和之前几乎如出一辙:你还是“死”了,而且同样“死”的很快!

现在小伙伴们可能有点意识到了:不是我们钩子代码的问题,而是整个调用体系的问题!

你们真是冰雪聪明!

不过,在揭秘真正的问题之前让我们先“站在巨人肩膀之上”来尝试另一种完全不同的解决方案吧!

在下一篇博文中,我们将再接再厉用第三方 Hook 包 SwiftHook 来重新探寻一番,期待吧!

总结

在本篇博文中,我们讨论了为什么要在 App 运行时“捕获”所有 NSObject 对象的实例、介绍了 NSObject 默认构造器方法里都做了神马事情,以及初步探讨了实现这一目的的基本原理。

虽然,眼前的团团迷雾让我们暂时还无法得偿所愿,但相信只要怀有坚定的信念,胜利就在眼前!

让我们在下篇继续跟随初心,拨开迷雾见青天吧!再会!😎

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/758608.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

有了std::thread,为什么还需要引入std::jthread?

C进阶专栏:http://t.csdnimg.cn/HGkeZ 目录 1.前言 2.std::is_invocable_v 3.std::jthread 3.1.构造函数 3.2.std::jthread无需join/detach使用实例 3.3.std::jthread处理外部请求中断实 3.4.处理中断请求示例代码 4.特性 5.总结 1.前言 C11以来提供了C原…

Redis实现高可用方案

文章目录 前言一、主从模式1.1 复制流程1.2 优缺点 二、哨兵模式2.1 介绍2.2 哨兵的工作模式 三、集群模式3.1 Cluster集群节点的通讯3.2 Hash Slot插槽算法3.3 Redis Cluster集群3.4 故障转移 前言 如果单机部署Redis服务的话,一旦Reids宕机,那么整个服…

英伟达 V100、A100/800、H100/800 GPU 对比

近期,不论是国外的 ChatGPT,还是国内诸多的大模型,让 AIGC 的市场一片爆火。而在 AIGC 的种种智能表现背后,均来自于堪称天文数字的算力支持。以 ChatGPT 为例,据微软高管透露,为 ChatGPT 提供算力支持的 A…

centos 环境部署

一、安装redis 1. 升级 GCC 最直接的解决方式是升级你的 GCC 编译器到支持 C11 标准的版本。CentOS 7 默认的 GCC 版本较旧,可能不支持 _Atomic。你可以通过以下步骤升级 GCC: 启用 CentOS 的 Software Collections (SCL) 仓库,该仓库提供了…

王老吉药业开拓数字经济“新蓝海”,成立数字经济研究所,科技赋能新品压片糖

3月12日,广州王老吉药业股份有限公司(以下简称“王老吉药业”)召开第十一届312感恩活动新闻发布会,宣告王老吉数字经济研究所成立,并发布王老吉压片糖新品。一系列重要重要举措,无一不标志着王老吉药业正以…

Java SE入门及基础(44)

目录 I / O流(上) 1. 什么是I / O流 过程分析 I / O的来源 Java 中的 I / O流 2. 字节流 OutputStream 常用方法 文件输出流 FileOutputStream 构造方法 示例 InputStream 常用方法 文件输入流 FileInputStream 构造方法 示例 综合练习 字节流应用场景 Java SE文…

自动化测试报告生成(Allure)

🍅 视频学习:文末有免费的配套视频可观看 🍅 关注公众号【互联网杂货铺】,回复 1 ,免费获取软件测试全套资料,资料在手,涨薪更快 之前尝试使用过testNG自带的测试报告、优化过reportNG的测试报告…

算法·动态规划Dynamic Programming

很多人听到动态规划或者什么dp数组了,或者是做到一道关于动态规划的题目时,就会有一种他很难且不好解决的恐惧心理,但是如果我们从基础的题目开始深入挖掘动规思想,在后边遇到动态规划的难题时就迎难而解了。  其实不然&#xff…

linux:线程互斥

个人主页 : 个人主页 个人专栏 : 《数据结构》 《C语言》《C》《Linux》 文章目录 前言一、线程互斥问题解释互斥量的接口 二、加锁的原理三、 死锁死锁四个必要条件避免死锁 总结 前言 本文是对于线程互斥的知识总结 一、线程互斥 问题 我们先看下面…

财报解读:出海“窗口期”再现,汇量科技保驾护航的底气源于什么

大数据时代,每个人的喜好都被精准捕捉。购物APP、购物网站们,都仿佛一位贴心的时尚顾问。而这源于个性化广告经过深度学习和智能算法得来的结果。 随着广告市场的竞争愈演愈烈,广告主们需要更为精准、高效的个性化投放。近日,深耕…

基于SSM的宿舍管理系统的设计与实现(JSP,MySQL)

摘 要 随着社会发展、信息技术的普及,人们日常管理工作也发生了巨大的变化。信息化技术之渗透各行业的方方面面。学生宿舍管理作为校园管理工作的重要一环,不仅关系到学生自身的确切利益,同时也是对校园管理工作重大考验。近来年由于在校学生…

leetcode代码记录(移除链表元素

目录 1. 题目:2. 我的代码:小结: 1. 题目: 给你一个链表的头节点 head 和一个整数 val ,请你删除链表中所有满足 Node.val val 的节点,并返回 新的头节点 。 示例 1: 输入:head […

Flutter开发进阶之瞧瞧Widget

Flutter开发进阶之瞧瞧Widget 在Flutter开发中,WIdget是构建界面的基本单元;Widget是不可变的,意味着一旦创建如果需要改变UI就需要重新创建一个新的Widget;在实际开发中,Widget通常由一个个Widget组合而成,从而形成嵌套的树形结构,复杂的UI就是由这一个个Widget构建而…

【C语言】—— 指针三 : 参透数组传参的本质

【C语言】—— 指针三 : 参透数组传参的本质 一、数组名的理解二、使用指针访问数组2.1、指针访问数组2.2、[ ] 的深入理解2.3、数组与指针的区别 三、一维数组的传参本质四、数组指针变量4.1、数组指针变量是什么4.2、 数组指针的初始化 五、二维数组传参的本质 一…

简单了解多线程

并发和并行 并发: 在同一时刻,多个指令在单一CPU上交替指向 并行:在同一时刻,多个指令在多个CPU上同时执行 2核4线程,4核8线程,8核16线程,16核32线程 基础实现线程的方式 Thread :继承类 &…

多人命题系统|基于SSM框架+ Mysql+Java+ B/S结构的多人命题系统设计与实现(可运行源码+数据库+设计文档)

推荐阅读100套最新项目 最新ssmjava项目文档视频演示可运行源码分享 最新jspjava项目文档视频演示可运行源码分享 最新Spring Boot项目文档视频演示可运行源码分享 2024年56套包含java,ssm,springboot的平台设计与实现项目系统开发资源(可…

13年资深测试,性能测试常见指标分析总结,看这篇就够了...

目录:导读 前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结(尾部小惊喜) 前言 1、什么是性能测试…

Kotlin runBlocking CoroutineScope synchronized简单死锁场景

Kotlin runBlocking CoroutineScope synchronized简单死锁场景 import kotlinx.coroutines.*fun main(args: Array<String>) {runBlocking {val lock1 Any()val lock2 Any()CoroutineScope(Dispatchers.IO).launch {repeat(10) {println("A-$it 申请 lock1...&quo…

Http 超文本传输协议基本概念学习摘录

目录 HTTP协议 超文本传输协议 HyperText超文本 HTML超文本标记语言 HTTP协议原理 请求发送 服务器处理 响应发送 连接关闭或保持 HTTP协议版本 HTTP/0.9 HTTP/1.0 HTTP/1.1 HTTP/2 HTTP/3 HTTP请求方法 GET POST PUT DELETE HEAD OPTIONS HTTP请求头字…

JVM学习-类加载

目录 1.类文件结构 2.类加载器 3.类加载的三个阶段 3.1加载 3.2链接 3.2.1验证 3.2.2准备阶段 3.2.3解析阶段 3.3初始化 4.拓展&#xff1a;反射 4.1获取类对象 4.2创建实例 4.3获取方法 4.4方法调用 1.类文件结构 2.类加载器 类加载器用来将类文件的二进制字节码加载到JV…