Hello World背后的逻辑

news2025/1/22 17:41:56

一门语言的开发入门,总是抬手就能整出一个「Hello World Demo」。比如下面这样:

image.png

显然,熟悉 iOS 开发的同学都知道,上面这个来自 Objective-C。

今天,我们就从这熟悉的代码入手,来一起研究研究「Hello World」出世的整个过程。

main 函数

众说周知,main 函数是我们程序的入口,我们不妨从此入手,开始我们的表演。

入手的姿势已经确定,甩手一个断点,拿到下图:

image.png 显然,在main函数执行之前,先是调用了start方法,那这个所谓的start方法又是什么呢?哪里来的呢?又是怎么调起来的呢?

从上图我们并不看的很真切,因+ (void)load {}方法的调用是在加载阶段(后面验证),而加载完之后才触发main函数的调用,所以我们在load方法时候再加上一个断点(这里可以随便弄个类,重写load方法),看看究竟。

+load方法

再次运行代码之后很容易先确认前面提到的「load 方法调用在 main 调用前」,并拿到下面的调用堆栈(bt命令可打印更详细的堆栈信息):

image.png 从上图我们可以清楚的看到一切的开始源于一堆dyld的东西,那么,dyld是个啥?

dyld(全名 the dynamic link editor)是苹果的动态链接器,用来链接所有的库和可执行文件,是苹果操作系统一个重要组成部分,在系统内核做好程序准备工作之后,交由 dyld 负责余下的工作。它的代码也是开源的,正常可以从这里下载

!!注意:网上有很多关于dyld执行流程的介绍,但都是基于稍老的一些版本,可以看到上面的堆栈信息与老版本的也有些许差异,但是总体流程上基本一致,这里介绍基于最新的dyld4版本,源码可从这里下载

我们接着说,在dyld做完加载库、可执行文件等一系列准备工作之后,通过dyld4::RuntimeState::notifyObjCInit 触发libobjc.A.dylib中的load_images函数,再到我们自定义的[Person load]方法的调用,最终到在之后的main函数。

此话怎么讲呢?iOS开发者对notify这样的字眼是不是有些熟悉?对的,通知!还是通知的触发,上面的过程很显然就是dyld触发通知到注册通知的接收方,哪里呢?libobjc.A.dylib中的load_images函数,这个库其实就是我们常念叨的RuntimeRuntime的代码也是在官方的开源库中,所以我们接下来可以直接验证一下我们的猜想。Runtime源码这里下载

这里额外说明一下,Runtime源码是可以运行起来的,当然需要一堆配置,这里提供一个可以直接跑起来的源码,在这里,下载下来之后,Target 选择KCObjcBuild就可以,源码中直接Debug,不要太爽。

load_images函数

哦了,接着说,我们怎么验证呢?直接在Runtime源码中直接搜索load_images,很容易定位到下面这里:

image.png 显然是load_images定义的地方,直接甩一断点验证看看,(源码直接编译优势凸显):

image.png 这里调用的堆栈信息,很显然符合我们的猜想。

继续,是通知就该有注册的地方,不然怎么就能调用到上面的load_images函数呢?前面搜索load_images的时候其实就有这么个地儿:

image.png

上图中我们看到了什么呢?是的,_dyld_objc_notify_register,盲猜一下,应该就是通知的注册地了吧。

既然如此,我们知道要想实现上面的调用过程,那么,通知的注册应该要在调用触发之前,不然可说不过去,来来来,接着验证,甩手一个断点(就问源码直接debug爽还是爽?):

image.png 果然如此吧,甚至上面的堆栈信息直接能看出来_objc_init的调用过程,算是意外之喜了。

_objc_init函数

_objc_init的调用过程,从上面的堆栈信息可以看到:

-> dyld`start 
-> dyld`dyld4::prepare() 
-> dyld`dyld4::APIs::runAllInitializersForMain() 
-> dyld`dyld4::Loader::findAndRunAllInitializers() 
-> dyld`dyld3::MachOAnalyzer::forEachInitializer() 
-> dyld`dyld3::MachOFile::forEachSection() const 
-> dyld`dyld3::MachOFile::forEachLoadCommand() 
-> dyld3::MachOFile::forEachSection() 
-> dyld3::MachOAnalyzer::forEachInitializer() 
-> dyld4::Loader::findAndRunAllInitializers() 
-> libSystem.B.dylib`libSystem_initializer 
-> libdispatch.dylib`libdispatch_init 
-> libdispatch.dylib`_os_object_init 
-> libobjc.A.dylib`_objc_init  

dyld相关的函数在dyld开源库中,前面已经提到了,这里。

libSystem.B.dylib的相关代码在官方的开源库中能直接拿到,这里。

libdispatch.dylib相关的代码也在官方的开源库中能查询到,这里。

libobjc.A.dylib就是上面提到的Runtime代码了,这里

运行轨迹

截止目前,我们大致了解了main函数调用的基本逻辑:

dyldstart开始 -> _objc_init函数加载(注册了load_images) -> 触发load_images函数 -> 触发+load方法 -> 在最后才调用main函数 -> 最终输出Hello World

可见,在main函数调用之前,确实还是有一大堆我们并没有意识到的操作,大部分是dyld在处理,我们姑且称之为–加载过程

但是main函数究竟是怎么被唤起的呢?目前看着仍旧不是很明朗,还是要继续撸源码,那块儿的源码呢?从前面的堆栈来看,还是要从dyldstartprepare入手。

dyld唤起main函数

我们可以在dyld源码中全局搜索prepare(,最终找到能这个地方:

image.png 而这里正好是start函数的内部调用,符合我们前面的堆栈信息,地方应该可以确认了,没跑了。

其实从这里我们就可以在此大胆假设,然后去小心求证了:

// load all dependents of program and bind them together
MainFunc appMain = prepare(state, dyldMA);

// now make all dyld Allocated data structures read-only
state.decWritable();

// call main() and if it returns, call exit() with the result
// Note: this is organized so that a backtrace in a program's main thread shows just "start" below "main"
int result = appMain(state.config.process.argc, state.config.process.argv, state.config.process.envp, state.config.process.apple);

从上面的命名及注释,很容易看到:prepare()主要就是处理main函数调用前的准备工作,比如加载所有的dependents,最终返回的就是main函数的入口(MainFunc类型),而后的appMain()实际上就是对main函数的调用了,我们可以看到其参数就跟我们main的参数神似了。

上面的这些,我们如果深入prepare()就会进一步得到验证,截取部分代码如下:

...
    // run all initializers
    state.runAllInitializersForMain();

    // notify we are about to call main
    notifyMonitoringDyldMain();
    if ( dyld3::kdebug_trace_dyld_enabled(DBG_DYLD_TIMING_LAUNCH_EXECUTABLE) ) {
        dyld3::kdebug_trace_dyld_duration_end(launchTraceID, DBG_DYLD_TIMING_LAUNCH_EXECUTABLE, 0, 0, 4);
    }
    ARIADNEDBG_CODE(220, 1);

    MainFunc result;
    if ( state.config.security.skipMain ) {
        return &fake_main;
    }
    else if ( state.config.process.platform == dyld3::Platform::driverKit ) {
        result = state.mainFunc();
        if ( result == 0 )
            halt("DriverKit main entry point not set");
#if __has_feature(ptrauth_calls)
        // HACK: DriverKit signs the pointer with a diversity different than dyld expects when calling the pointer.
        result = (MainFunc)__builtin_ptrauth_strip((void*)result, ptrauth_key_function_pointer);
        result = (MainFunc)__builtin_ptrauth_sign_unauthenticated((void*)result, 0, 0);
#endif
    }
    else {
        // find entry point for main executable
        uint64_t entryOffset;
        bool     usesCRT;
        if ( !state.config.process.mainExecutable->getEntry(entryOffset, usesCRT) )
            halt("main executable has no entry point");
        result = (MainFunc)((uintptr_t)state.config.process.mainExecutable + entryOffset);
        if ( usesCRT ) {
            // main executable uses LC_UNIXTHREAD, dyld needs to cut back kernel arg stack and jump to "start"
#if SUPPPORT_PRE_LC_MAIN
            // backsolve for KernelArgs (original stack entry point in _dyld_start)
            const KernelArgs* kernArgs = (KernelArgs*)(&state.config.process.argv[-2]);
            gotoAppStart((uintptr_t)result, kernArgs);
#else
            halt("main executable is missing LC_MAIN");
#endif
        }
#if __has_feature(ptrauth_calls)
        result = (MainFunc)__builtin_ptrauth_sign_unauthenticated((void*)result, 0, 0);
#endif
    }

runAllInitializersForMain()也能在前面的堆栈中得以体现,函数名也很直观,为main函数的调用,做所有的初始化工作。

而下面的result赋值的过程,实际上就是main函数入口查找的过程。

总结

综上所述,main函数调用前的过程我们更清晰了,一切的开始是从dyldstart函数开始,其通过prepare()函数做好了main函数调用前的所有准备工作,如初始化、链接所有的动态库及可执行文件等,查找main函数的入口并返回到start函数后,实现main函数的调用触发。

当然,这中间实际上还有一些模糊的地方,比如_objc_init里面具体干了啥,load_images有什么用,以及上面的appMain就是我们最开始截图里面的main方法吗?(显然不是,参数数量对不上,哈哈,中间还有一些过程)。

有兴趣的小伙伴可以继续研究研究,后续我们再继续讨论。

Python 的迅速崛起对整个行业来说都是极其有利的 ,但“人红是非多”,导致它平添了许许多多的批评,不过依旧挡不住它火爆的发展势头。

如果你对Python感兴趣,想要学习python,这里给大家分享一份Python全套学习资料,都是我自己学习时整理的,希望可以帮到你,一起加油!

😝有需要的小伙伴,可以点击下方链接免费领取或者V扫描下方二维码免费领取🆓
Python全套学习资料

在这里插入图片描述

1️⃣零基础入门

① 学习路线

对于从来没有接触过Python的同学,我们帮你准备了详细的学习成长路线图。可以说是最科学最系统的学习路线,你可以按照上面的知识点去找对应的学习资源,保证自己学得较为全面。
在这里插入图片描述

② 路线对应学习视频

还有很多适合0基础入门的学习视频,有了这些视频,轻轻松松上手Python~
在这里插入图片描述

③练习题

每节视频课后,都有对应的练习题哦,可以检验学习成果哈哈!
在这里插入图片描述

2️⃣国内外Python书籍、文档

① 文档和书籍资料

在这里插入图片描述

3️⃣Python工具包+项目源码合集

①Python工具包

学习Python常用的开发软件都在这里了!每个都有详细的安装教程,保证你可以安装成功哦!
在这里插入图片描述

②Python实战案例

光学理论是没用的,要学会跟着一起敲代码,动手实操,才能将自己的所学运用到实际当中去,这时候可以搞点实战案例来学习。100+实战案例源码等你来拿!
在这里插入图片描述

③Python小游戏源码

如果觉得上面的实战案例有点枯燥,可以试试自己用Python编写小游戏,让你的学习过程中增添一点趣味!
在这里插入图片描述

4️⃣Python面试题

我们学会了Python之后,有了技能就可以出去找工作啦!下面这些面试题是都来自阿里、腾讯、字节等一线互联网大厂,并且有阿里大佬给出了权威的解答,刷完这一套面试资料相信大家都能找到满意的工作。
在这里插入图片描述
在这里插入图片描述

5️⃣Python兼职渠道

而且学会Python以后,还可以在各大兼职平台接单赚钱,各种兼职渠道+兼职注意事项+如何和客户沟通,我都整理成文档了。
在这里插入图片描述

上述所有资料 ⚡️ ,朋友们如果有需要的,可以扫描下方👇👇👇二维码免费领取🆓
在这里插入图片描述

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

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

相关文章

泄露35TB数据,医疗巨头Henry Schein遭受黑猫勒索组织攻击

近日,据Bleeping Computer 网站消息,BlackCat(黑猫)勒索软件团伙将医疗保健巨头Henry Schein 添加到了其暗网泄露网站,并声称其破坏了该公司的网络,窃取了35 TB的敏感文件,这些文件包括了Henry …

BES 在大规模向量数据库场景的探索和实践

导读 本文整理自 2023 年 9 月 5 日 QCon 全球软件开发大会 2023 北京站 —— 向量数据库分论坛的同名主题演讲《BES 在大规模向量数据库场景的探索和实践》。 全文5989字,预计阅读时间15分钟。 向量数据库是一种专门用于存储和查询向量数据的数据库系统。通过 Emb…

verdi如何打开时可以加载配置比如字体

打开tcl使能 找到配置字体的命令 其实其他有需要的文件配置都可以在这里找到对应的指令 存储文件 新建verdi001.tcl文件 输入想要调整的字体以及大小 verdiSetFont -font "Bitstream Vera Sans" -size "18" verdiSetFont -monoFont "Courier&q…

【 云原生 | K8S 】kubectl 详解

目录 1 kubectl 2 基本信息查看 2.1 查看 master 节点状态 2.2 查看命名空间 2.3 查看default命名空间的所有资源 2.4 创建命名空间app 2.5 删除命名空间app 2.6 在命名空间kube-public 创建副本控制器(deployment)来启动Pod(nginx-wl…

做哪些副业可以日赚一百?对程序员来说简直不要太容易!

日赚一百?对程序员来说简直不要太容易!下面给程序员们推荐一些日赚100的副业: ①外包接单 程序员简单粗暴赚钱的副业之一。 外包接单的类型包括但不限于:软件开发、硬件开发、小程序功能开发、web开发……大到一个系统的开发、…

什么样的CRM系统更适合外贸企业?

外贸CRM系统作为外贸客户关系管理的工具,已经成为了当下外贸企业对外贸易过程中不可或缺的一环。那什么样的CRM系统更适合外贸企业?小Z向您推荐Zoho CRM。下面说说它到底有什么好处和作用。 一、搭建更高效的客户关系管理系统 外贸企业从前期推广、开发…

202205(第13届)蓝桥杯Scratch图形化编程青少组(国赛_中级)真题

202205(第13届)蓝桥杯Scratch图形化编程青少组(国赛_中级)真题 第 1 题 以下程序,小猫在移动完成后不能回到初始位置的是?( ) A: B: C: D: 第 2 题 以下程序,询问…

SAP 10策略测试及简介

从今天开始将把PP模块中常用的一些策略进行一个测试,编写成系统的文档,有点策略经常不用自己都忘了一些策略的特性。所以还是有必须形成文档的形式记录下来 1、首先准备好物料 成品物料为AB0,在MRP3视图中维护对应的策略组的10 同时选择消耗模式为2.消耗期间都是999 2、其他…

环境变量小结

一 常见环境变量介绍 1 PATH 到了现在,我们也知道我们轻轻敲下ls指令,其实会转为一个可执行文件在运行,也就变成了一个进程,所以ls是让文件运行,./test也是让文件运行,凭什么我们的可执行文件就要加个./(这…

IDEA调试总结

前言 由于 IDEA 每个人使用的版本不同以及快捷键的设置不同,所以忽略了快捷键的使用。如果不知道快捷键请在 IDEA 工具栏里面点开 Run 菜单即可知悉 图标介绍 下面咱们进入看图说话环节,下列图标小伙伴知道是啥功能么?日常开发进行 Debug 使…

Spring-Security前后端分离权限认证

前后端分离 一般来说,我们用SpringSecurity默认的话是前后端整在一起的,比如thymeleaf或者Freemarker,SpringSecurity还自带login登录页,还让你配置登出页,错误页。 但是现在前后端分离才是正道,前后端分离的话,那就…

React状态管理方案盘点

您好, 如果喜欢我的文章或者想上岸大厂,可以关注公众号「量子前端」,将不定期关注推送前端好文、分享就业资料秘籍,也希望有机会一对一帮助你实现梦想 前言 本文不会介绍各个状态管理工具的具体使用或者如何二次封装&#xff0c…

自动驾驶系统激光雷达传感器反射率标定板

自动驾驶技术正在全球范围内快速发展和推广。在中国,自动驾驶技术也得到了高度重视和大力支持。中国政府已经出台了一系列政策,推动自动驾驶技术的发展和应用。例如,上海、北京等地已经开放了自动驾驶测试道路,并开展了自动驾驶公…

选择CRM系统主要看哪些指标?

很多企业都想选择一款好用的CRM客户管理系统,但是面对众多类型、品牌的CRM却犯了难。下面我们来说说,企业要想选到一款适合自己的、好用的CRM系统,主要看哪些指标?这里有6个步骤,可以帮您做到。 第1步:了解…

亚马逊鲲鹏系统六大优势

亚马逊鲲鹏系统六大优势凭借其独特的能力,完全模拟真实的人类行为。只需几个简单的步骤 就可以自由安排任务,让所有账户随时发挥最大的作用。 1、全自动化操作 可以全自动批量注册买家号、AI智能养号、全自动批量测评,模拟人类的操作行为例…

亚马逊鲲鹏系统能做什么

亚马逊鲲鹏系统是一款能绕过亚马逊智能检测,完全模拟人类真实行为,通过模拟真实的人流量来帮助你提升你的产品排名,让你的产品出现在搜索首页,从而快速提高你的销售业绩的营销工具! 主要的功能有批量注册买家号、AI智能…

微服务概念

微服务 微服务是什么 In short, the microservice architectural style [1] is an approach to developing a single application as a suite of small services, each running in its own process and communicating with lightweight mechanisms, often an HTTP resource A…

node插件MongoDB(四)—— 库mongoose 的文档操作使用

文章目录 前言(1)问题:安装的mongoose 库版本不应该过高导致的问题(2)重新安装低版本 一、插入文档1. 代码2. node终端效果3. 使用mongo.exe查询数据库的内容 二、删除文档1. 删除一条2. 批量删除3. 代码 前言 &#…

新方向!文心一言X具身智能,用LLM大模型驱动智能小车

具身智能已成为近年来研究的热点领域之一。具身智能强调将智能体与实体环境相结合,通过智能体与环境的交互,来感知和理解世界,最终实现在真实环境中的自主决策和运动控制。 如何基于文心大模型,低成本入门“具身智能”&#xff0…

社区团购小程序系统源码+各种快递代收+社区便利店 带完整的搭建教程

社区团购小程序系统源码的开发背景可以追溯到近年来电商行业的快速发展,特别是在新冠疫情的影响下,线上购物在全球范围内得到了更广泛的普及和使用。社区团购作为一种新兴的电商模式,结合了传统团购和社交电商的优点,通过线上平台…