iOS 动态库 & 静态库

编译器

LLVM 是编译器工具链技术的一个集合。而其中的 lld 项目,就是内置链接器。 编译器会对每个文件进行编译,生成 Mach-O(可执行文件);链接器会将项目中的多个 Mach-O 文件合并成一个。

主要过程:

  • 预处理,比如把宏嵌入到对应的位置
  • 词法分析和语法分析,生成 AST(抽象语法树,结构上比代码更精简,遍历起来更快)
  • AST 生成 IR(一种更接近机器码的语言),iOS里IR 生成的可执行文件 Mach-O。

解释器:
在运行时才去解析代码,比在运行之前通过编译器生成一份完整的机器码再去执行的效率要低。具有动态性,程序运行后能够随时通过增加和更新 代码来改变程序的逻辑。

编译过程

链接器

Mach-O 文件里面的内容,主要就是代码和数据。它们的实例都需要由符号将其关联起来。完成变量、函数符号和其地址绑定这样的任务。

链接器在链接多个目标文件的过程中,会创建一个符号表,用于记录所有已定义的和所有未定义的符号。

主要流程:

  • 查找目标代码文件里没有定义的变量
  • 将所有符号定义和引用地址收集起来,并放到全局符号表中
  • 计算合并后长度及位置,生成同类型的段进行合并,建立绑定
  • 对项目中不同文件里的变量进行地址重定位
image.png

库是共享程序代码的方式。库从本质是一种可执行代码的二进制格式,可以被载入内存中执行。出于安全性和稳定性的考虑,不想被外界知道,所以会把核心代码打包成库,只暴露出头文件以供使用。

静态库 & 动态库

静态库 & 动态库

静态库

  • 是编译时链接的库,需要链接进 Mach-O 文 件里,如果需要更新就要重新编译一次,无法动态加载和更新。

  • 链接时会被完整的复制到可执行文件中,所以如果两个程序都用了某个静态库,那么每个二进制可执行文件里面其实都含有这份静态库的代码。

静态库

动态库

  • 运行时链接的库, 使用 dyld 就可以实现动态加载。没参与 Mach-O 文件的编译和链接。Mach-O 文件中并没有包含动态库里的符号定义,“未定义”。名字和对应的库的路径会被记录。

  • 运行时通过 dlopen 和 dlsym 导入动态库时,先根据记录的库路径找到对应的库,再通过记录的名字符号找到绑定的地址。

  • 链接时不复制,程序运行时由系统动态加载到内存,供程序调用,系统只加载一次,多个程序共用,节省内存。

动态库

dyld 加载动态库

启动加载时绑定和符号第一次被用到时绑定。


image.png

iOS 动态库

iOS 8 之后,iOS有了App Extesion特性;由于 iOS 主App需要和Extension共享代码,Swift语言机制也需要动态库,于是苹果后来提出了Embedded Framework,这种动态库允许APP和APP Extension共享代码,但是这份动态库的生命被限定在一个APP进程内。

App Extesion(应用扩展)

https://www.jianshu.com/p/fbab922fc175

不是一个app,而只是对于某个app内容和功能的扩展。跟随着你的app一起打包,包含应用扩展一起打包的app就叫做container app(容器app)。
运行时它并不是跟你的app在同一个进程上面,而且有可能同一个app extension会同时运行在不同的进程,因为有可能同时有几个程序都打开了这个app extension,这个用来打开某个app Extension的应用就叫做host app(宿主应用)。

  • Call Directory用来处理来电时的事项,比如黑名单机制、来电标记提醒等等
  • Photo Editing用来在相册选中相片编辑时的扩展,可以直接对相片进行一个编辑处理
  • Share用来对文件进行一个直接的分享功能,他不用唤起主App就能完成文件的分享

系统动态库和自定义动态库(Embedded Framework)区别

  • 传统的动态库(UIKit.Framework)是给多个进程用的
  • 动态库(Embedded Framework)是给单个进程里面多个可执行文件用的。
  • 系统的 Framework 不需要拷贝到目标程序中
  • 自定义动态库哪怕是动态的,最后也还是要拷贝到 App 中

什么是framework

Framework是Cocoa/Cocoa Touch程序中使用的一种资源打包方式,可以将代码文件、头文件、资源文件、说明文档等集中在一起。


framework

framwork 目录

目录
  • Headers 表示暴露的头文件


    headers
  • info.plist 一些配置信息

  • Modules
    Headers中暴露的XXXX.h文件被放在umbrella下保护起来了,所以我们需要将其他的所有需要暴露的.h文件放到XXXX.h文件中保护起来
    .swiftmodule 文件
    因为swift没有头文件,.swiftmodule文件会包含AST等信息,会充当头文件的作用。


    swiftmodule
  • .bundle文件
    Build Phases -> Copy Bundle Resources


    bundle
bundle

framework 资源文件

工程截图
输出文件

一般如果是静态Framework的话,资源打包进Framework是读取不了的。静态Framework和.a文件都是编译进可执行文件里面的。
只有动态Framework能在.app的Framework文件夹下看到,并读取.framework里的资源文件。

let enBundle = Bundle.init(path: bundle.path(forResource: "xxxx", ofType: "xxxx") ?? "") 

静态库和动态库依赖关系

1. 静态库互相依赖

制作静态库的时候只需要有被依赖的静态库头文件在就能编译出来。

2. 动态库依赖动态库

两个动态库是相互隔离的具有隔离性。在制作的静态库的时候需要被依赖动态库参与链接,最终具体的符号决议交给dyld来做。

3. 静态库依赖动态库

静态库制作的时候也需要动态库参与链接,但是符号的决议交给dyld来做。

4. 动态库依赖静态库

静态库本质是一堆.o 的打包体,首先并不是二进制可执行文件,再者你无法保证主程序把静态库参与链接共同生成二进制可执行文件。

解决方案
目前的编译器的解决办法是,首先我无法保证主程序是否包含静态库,再者静态库也无法被dyld加载,那么我直接把你静态库的.o 偷过来,共同组成一个新的二进制。也被称做吸附性。

【如果多个动态库依赖同一静态库?】
可执文件(主程序或者动态库)在构建的链接阶段,遇到静态库,吸附进来;遇到动态库,打标记,彼此保持独立。

正因为动态库是保持独立的,可以自定义一个动态库把依赖的静态库吸附进来。对外整体呈现的是动态库特性。其他的组件依赖我们自定义的动态库,由于隔离性的存在,不会出现问题。

注意事项

静态库中有category类

【Other Linker Flags】需要添加参数【-ObjC]或者【-all_load】

【dyld: Library not loaded:XXXXXX】

打包的framework版本,需要到iOS Deployment Target设置对应版本。

Include of non-modular header inside framework module

framework类中使用了.dylib或者.tbd,首先需要在实际项目中导入.dylib或者.tbd动态库,然后需要设置【Allow Non-modular Includes ....】为YES

【Umbrella header for module 'XXXX' does not include header 'XXXXX.h'】

错把xxxxx.h拖到了public中

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 221,548评论 6 515
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 94,497评论 3 399
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 167,990评论 0 360
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 59,618评论 1 296
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 68,618评论 6 397
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 52,246评论 1 308
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,819评论 3 421
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,725评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 46,268评论 1 320
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,356评论 3 340
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,488评论 1 352
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 36,181评论 5 350
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,862评论 3 333
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,331评论 0 24
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,445评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,897评论 3 376
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,500评论 2 359

推荐阅读更多精彩内容