库
- 库是一份可执行代码的二进制代码,可以被操作系统载入内存,并且被识别、使用。
- 在链接阶段,会将汇编生成的目标文件与引用的库一起链接,打包进可执行文件中。
分类
静态库
- 以
.a
、.framework
为后缀的文件。 - 静态库在链接时会被完整的复制到可执行文件中,当静态库被多次使用时,会进行多次复制,从而出现拷贝冗余,造成内存浪费。
- 优点:不受外部环境的影响,即使删除了静态库,对可执行文件不会造成影响,因为静态库在链接时就已经打包到了可执行文件中,成为 App 的一部分。
- 缺点:浪费内存空间。如果静态库进行了修改,可执行文件也需要重新编译生成。
注意:
.a
是一个纯二进制文件,而.framework
中除了有二进制文件之外还有资源文件。
动态库
- 以
.tbd
、.framework
、.xcframework
为后缀的文件。 - 动态库在链接时不会直接复制,而只会存储指向动态库的引用,等到程序运行时才被载入到内存中,以供使用。
- 优点:只加载到内存中一次,内存共享,节约内存空间。可以独立于 App 进行更新,因为它并不是 App 的一部分。
- 缺点:运行时载入会造成性能损失,而且可执行文件依赖外部环境,一旦动态库进行了修改而出现了错误,则会导致程序出现问题。
如何区分
- 查看 Build Settings —> Mach-O Type 的值。
- 查看 ipa 的目录结构。
- 通过
file xxx.a/framework
命令查看。
混合使用
- 静态库可以依赖静态库。
- 动态库可以依赖动态库。
- 动态库不能依赖静态库。原因是静态库不需要在运行时再次加载, 如果多个动态库依赖同一个静态库, 则会出现多个静态库的拷贝。
framework
- 一种资源打包方式,可以将代码文件、头文件、资源文件、文档等集中在一起。
- 静态
framework
:打包成 SDK 供别人使用的一般都是静态framework
。 - 动态
framework
:系统提供的framework
是动态的。例如:Foundation.framework
、UIKit.framework
等。 - 开发者也可以制作的动态
framework
,但会受到平台限制,需要通过Embedded Framework
的方式进行工作,与系统动态库不同,这种动态库最终也要拷贝到 App 中。
Frameworks,Libraries,and Embedded Content
- 对于系统动态库,可以将
Embed
属性设置成Do Not Embed
。 - 对于用户动态库,需要将
Embed
属性设置成Embed
。 - 对于静态库,需要将
Embed
属性设置成Do Not Embed
。
创建
- Xcode 可以直接创建 Static Library 静态库,最终生成
.a
文件。 - Xcode 也可以直接创建 Framework,创建完成后通过 Build Settings —> Mach-O Type 调整为
Dynamic Library
或者Static Library
。
dyld
- The dynamic link editor(苹果动态链接器),是 iOS 中非常重要的组成部分。
- 程序启动时,通过
dyld
将库加载到内存。
优化
Xcode 15 之后可以进一步合并动静态库(mergeable libraries
),根据需要设置 Build Settings —> Create Merged Binary 对应的值即可。
参考
- Xcode 15 优化
- iOS整理: 关于动态库和静态库