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 in
        print(obj)
    } as @convention(block) (NSObject, Selector) -> Void
    
    let 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 in
    let 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 默认构造器方法里都做了神马事情,以及初步探讨了实现这一目的的基本原理。

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

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

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

leetcode代码记录(删除字符串中的所有相邻重复项

目录 1. 题目:2. 我的代码:小结: 1. 题目: 给出由小写字母组成的字符串 S,重复项删除操作会选择两个相邻且相同的字母,并删除它们。 在 S 上反复执行重复项删除操作,直到无法继续删除。 在完成…

Uibot6.0 (RPA财务机器人师资培训第1天 )RPA+AI、RPA基础语法

训练网站:泓江科技 (lessonplan.cn)https://laiye.lessonplan.cn/list/ec0f5080-e1de-11ee-a1d8-3f479df4d981(本博客中会有部分课程ppt截屏,如有侵权请及请及时与小北我取得联系~) 紧接着小北之前的几篇博客,友友们我们即将开展新课的学习~…

Socket类

2.2 Socket类 Socket 类:该类实现客户端套接字,套接字指的是两台设备之间通讯的端点。 构造方法 public Socket(String host, int port) :创建套接字对象并将其连接到指定主机上的指定端口号。如果指定的host是null ,则相当于指定地址为回送…

论文阅读之AN IMAGE IS WORTH 16X16 WORDS: TRANSFORMERS FOR IMAGE RECOGNITION AT SCALE

文章目录 原文链接主要内容模型图技术细节实验结果 原文链接 AN IMAGE IS WORTH 16X16 WORDS: TRANSFORMERS FOR IMAGE RECOGNITION AT SCALE 主要内容 这篇文章的主要内容是介绍了一种新的计算机视觉模型——Vision Transformer(ViT),这是…

爬虫基础:HTTP基本原理

爬虫基础:HTTP基本原理 前言HTTP基本原理URI 和 URLHTTP 和 HTTPSHTTP 请求过程请求与响应HTTP请求HTTP响应请求与响应的交互过程 HTTP 2.0二进制传输多路复用Header压缩服务器端提前响应内容安全 前言 了解 HTTP的基本原理,了解从往测览器中输人 URL到获…

软件安全测评要点有哪些?第三方软件安全测试是必需品吗?

在当今数字化时代,软件安全测试是每个软件开发团队都不能忽视的重要环节。安全测试是指对软件产品进行系统、全面的安全性评测与检测的过程。它旨在发现并修复软件中存在的漏洞和安全隐患,以确保软件能够在使用过程中保护用户的数据和隐私不被非法访问和…

Pytest自动化测试框架快速上手(超详细)

🍅 视频学习:文末有免费的配套视频可观看 🍅 关注公众号:互联网杂货铺,回复1 ,免费获取软件测试全套资料,资料在手,薪资嘎嘎涨 pytest是一个非常成熟的全功能的Python测试框架&#…

《操作系统实践-基于Linux应用与内核编程》第10章--实验 Qt聊天程序

前言: 内容参考《操作系统实践-基于Linux应用与内核编程》一书的示例代码和教材内容,所做的读书笔记。本文记录再这里按照书中示例做一遍代码编程实践加深对操作系统的理解。 引用: 《操作系统实践-基于Linux应用与内核编程》 作者:房胜、李旭健、黄…

边缘检测-Tiny and Efficient Model for the Edge Detection Generalization

源代码: https://github.com/xavysp/TEED 论文地址:https://arxiv.org/pdf/2308.06468.pdf 大多数高级计算机视觉任务依赖于低级图像操作作为其初始过程。边缘检测、图像增强和超分辨率等操作为更高级的图像分析提供了基础。在这项工作中,我们考虑三个…

YUNBEE云贝-OBCP大军又一满分学员登榜!

课程介绍 培训概述 OceanBase 认证是 OceanBase 官方推出的唯一人才能力认证体系,代表了阿里巴巴及蚂蚁集团官方对考生关于 OceanBase 技术能力的认可,旨在帮助考生更好地学习 OceanBase 数据库产品,早日融入 OceanBase 技术生态体系&#x…

浏览量这么低,还要不要继续坚持?

哈喽,你好啊,我是雷工! 曾经在一个群里聊天,有群友看到我两位数的浏览量,说到:浏览量这么低还坚持什么? 浏览量低是事实,大多数是十几二十的,上百的都是少数&#xff0c…

ubuntu10.04 apache2.2开启tls1.2的支持,使现代的edge和firefox浏览器能正常访问https

最近发现自己ubuntu10.04服务器上的apache https无法通过win11上的edge和firefox浏览器访问,但xp下的ie6和ie8没有问题。 firefox的错误提示为“此网站可能不支持TLS 1.2协议,而这是Firefox支持的最低版本”。 经过检查发现: IE6访问https所需的版本是SS…

Vulnhub靶机:Stapler1

一、介绍 运行环境:Virtualbox 攻击机:kali(10.0.2.4) 靶机:Stapler: 1(10.0.2.13) 目标:获取靶机root权限和flag 靶机下载地址:https://www.vulnhub.com/entry/sta…

安卓转鸿蒙能有多适配?简直了……

到现在为止,想必很多开发者都或多或少 了解过鸿蒙。许多企业也都已经加入了鸿蒙业务,半推半就的开始学习鸿蒙开发。那么鸿蒙到底好不好搞呢? 首先可以肯定的一点,对于做安卓的来说鸿蒙非常搞,究竟有多好搞呢&#xff…

蓝桥杯-体育健将-CPP-贪心

目录 一、题目描述: 二、整体思路: 三、代码: 一、题目描述: 二、整体思路: 要在k分钟内拿最多的金牌,就意味着要参加尽可能多的项目,因此就要选择耗时(比赛时间和休息时间)最少的项目先预处…

2024全国水科技大会【联合主办】福州水务集团有限公司

福州水务成立于2008年11月,AA信用评级,注册资本21.2亿元。下属各级企业70多家(包括3家国家级高新技术企业、1家A股上市企业)。集团主营供水、排水、环保、温泉文旅、综合服务五大板块,旗下运营自来水厂17座&#xff0c…

解锁数据价值:COS支持日志检索与分析功能

前言 腾讯云对象存储服务(COS)一直致力于为用户提供高效、安全、便捷的云存储服务。但是,当数据流动如同星辰大海,如何捕捉那些关键的瞬间,洞察每一次访问背后的故事?现在,日志检索与分析功能可…

Dijkstra算法

Dijkstra算法用于求无向图或者有向图中起点到各个顶点的最短路径,且边的权值需要为非负数下面这个题就可以用该算法求解 743. 网络延迟时间 有 n 个网络节点,标记为 1 到 n。 给你一个列表 times,表示信号经过 有向 边的传递时间。 times[i]…

MNN createSession 之 Schedule(三)

系列文章目录 MNN createFromBuffer(一) MNN createRuntime(二) MNN createSession 之 Schedule(三) MNN createSession 之创建流水线后端(四) MNN Session::resize 之流水线编码&am…

har的编译及引用

1.创建HAR 选择文件->新建->模块,然后再下一个页面选择static library,之后在接下来的页面设置模块名字,然后下一步直到完成。 2.创建成功后在新建的模块下编写自己的代码内容。 3.编译HAR 编译默认是从Index.ets文件下进行导出,如果…