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

本文主要是介绍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 默认构造器方法里都做了神马事情,以及初步探讨了实现这一目的的基本原理。

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

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

这篇关于Swift 从获取所有 NSObject 对象聊起:ObjC、汇编语言以及底层方法调用链(一)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


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

相关文章

C/C++和OpenCV实现调用摄像头

《C/C++和OpenCV实现调用摄像头》本文主要介绍了C/C++和OpenCV实现调用摄像头,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一... 目录准备工作1. 打开摄像头2. 读取视频帧3. 显示视频帧4. 释放资源5. 获取和设置摄像头属性

MySQL启动报错:InnoDB表空间丢失问题及解决方法

《MySQL启动报错:InnoDB表空间丢失问题及解决方法》在启动MySQL时,遇到了InnoDB:Tablespace5975wasnotfound,该错误表明MySQL在启动过程中无法找到指定的s... 目录mysql 启动报错:InnoDB 表空间丢失问题及解决方法错误分析解决方案1. 启用 inno

Python函数返回多个值的多种方法小结

《Python函数返回多个值的多种方法小结》在Python中,函数通常用于封装一段代码,使其可以重复调用,有时,我们希望一个函数能够返回多个值,Python提供了几种不同的方法来实现这一点,需要的朋友... 目录一、使用元组(Tuple):二、使用列表(list)三、使用字典(Dictionary)四、 使

Linux查看系统盘和SSD盘的容量、型号及挂载信息的方法

《Linux查看系统盘和SSD盘的容量、型号及挂载信息的方法》在Linux系统中,管理磁盘设备和分区是日常运维工作的重要部分,而lsblk命令是一个强大的工具,它用于列出系统中的块设备(blockde... 目录1. 查看所有磁盘的物理信息方法 1:使用 lsblk(推荐)方法 2:使用 fdisk -l(

使用Python获取JS加载的数据的多种实现方法

《使用Python获取JS加载的数据的多种实现方法》在当今的互联网时代,网页数据的动态加载已经成为一种常见的技术手段,许多现代网站通过JavaScript(JS)动态加载内容,这使得传统的静态网页爬取... 目录引言一、动态 网页与js加载数据的原理二、python爬取JS加载数据的方法(一)分析网络请求1

MySQL查看表的最后一个ID的常见方法

《MySQL查看表的最后一个ID的常见方法》在使用MySQL数据库时,我们经常会遇到需要查看表中最后一个id值的场景,无论是为了调试、数据分析还是其他用途,了解如何快速获取最后一个id都是非常实用的技... 目录背景介绍方法一:使用MAX()函数示例代码解释适用场景方法二:按id降序排序并取第一条示例代码解

Python中合并列表(list)的六种方法小结

《Python中合并列表(list)的六种方法小结》本文主要介绍了Python中合并列表(list)的六种方法小结,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋... 目录一、直接用 + 合并列表二、用 extend() js方法三、用 zip() 函数交叉合并四、用

Java 中的跨域问题解决方法

《Java中的跨域问题解决方法》跨域问题本质上是浏览器的一种安全机制,与Java本身无关,但Java后端开发者需要理解其来源以便正确解决,下面给大家介绍Java中的跨域问题解决方法,感兴趣的朋友一起... 目录1、Java 中跨域问题的来源1.1. 浏览器同源策略(Same-Origin Policy)1.

通过cmd获取网卡速率的代码

《通过cmd获取网卡速率的代码》今天从群里看到通过bat获取网卡速率两段代码,感觉还不错,学习bat的朋友可以参考一下... 1、本机有线网卡支持的最高速度:%v%@echo off & setlocal enabledelayedexpansionecho 代码开始echo 65001编码获取: >

Java Stream.reduce()方法操作实际案例讲解

《JavaStream.reduce()方法操作实际案例讲解》reduce是JavaStreamAPI中的一个核心操作,用于将流中的元素组合起来产生单个结果,:本文主要介绍JavaStream.... 目录一、reduce的基本概念1. 什么是reduce操作2. reduce方法的三种形式二、reduce