iOS中库的相关概念
- 库: 就是一段编译好的二进制代码,加上头文件就可以供别人使用了
- 库的分类:开源库和闭源库的形式
- 开源库如:AFNetworking, 源码一般放在某个版本控制库中
- 很多人直接下载开源代码,将相关文件copy到自己的工程,直接使用,缺点是当开源库的版本更新后,还要在手动copy一次,比较麻烦
- 使用git的submodule
- 现在比较常用的是cocoapods,用命令导入开源库,而且不用关心开源库的相关依赖。cocoapods的本质是将公用库编译成静态库,然后使得主工程依赖此静态库
- 开源库如:AFNetworking, 源码一般放在某个版本控制库中
- 闭源库
- 用途: 某些代码需要给别人使用,但是又不希望给被人看到源码,就需要以库的形式进行封装,只暴露出头文件
- 对某些不会进行大的改动的代码,我们想要减少编译的时间,就可以打包成库,因为库是已经编译好的二进制文件了,编译的时候Link一下就可以了,不会浪费编译时间。
- 根据使用库Link的方式,闭源库可以分为静态库和动态库
- 静态库:(.a或者.framework )在编译的时候会被直接copy一份,复制到目标程序里,在编译完成之后,库文件实际就没有多大作用了,目标程序没有外部依赖,可以直接运行,缺点是(目标程序的体积会增大)
- .a格式的静态库,一般使用的时候需要提供.h .a .bundle文件,由于创建静态库项目的时候编译出来的静态库只支持特定的一种硬件架构体系,所以如果需要生成通用静态库,需要用到lipo命令将多个静态库合并。
- .framework格式的静态库,framework最终只是一个bundle(一个文件夹,里面按照规定的目录结构方式文件),但是对XCode而言这样的target还是有真假(Real/Fake)之分,只有真的情况下XCode主工程在添加依赖的时候才能够选择此公用库项目的framework涨的target
- 动态库:在编译时不会被copy盗目标程序,目标程序只会存储指向动态库的引用,等到程序运行时,动态库才会被真正加载进来。不会影响目标程序的体积,但是会带来一部分性能损失。苹果不允许自制动态库上架到appstore
iOS中.a静态库的制作
-
新建XCode中的lib工程如下图所示:
-
将静态库作为子项目引用到其他工程
<p>注:引用静态库的工程的Target Dependencies和Link Binary With Libraries均需要添加.a静态库</p>
<p style ="text-indent:2em">引用静态库的工程,需要在Header Search Paths中添加静态库项目的头文件的路径(".."表示返回上层目录)</p>
<p style ="text-indent:2em">制作的静态库中如果添加了分类,引用静态库的工程需要在Other linker Flags中添加-ObjC</p>
- 编译静态库
-
编译静态库项目中的所有文件:
- 编译静态库项目中的部分文件:
[图片上传失败...(image-86e6c0-1519910623969)]
<p>注:当只需要打包部分文件时,只需要在打包静态库的工程里面不需要打包的.m文件的Taget MemberShip中的勾选去掉,则该文件将不会被打包到静态库中去</p>
<p style ="text-indent:2em">查看自己打包的静态库都包含了哪些文件可以用lipo命令</p>
- lipo 命令的用法
- 查看静态库中包含哪些架构
lipo -info lib.a
- 合并模拟器库文件和真机库文件
lipo -create -output lib.a lib-armv6.a lib-i386.a
- 解压出指定架构的静态库
lipo -extract_family armv7 -output lib-armv7.a lib.a
或者
lipo lib.a -thin armv7 -output lib-armv7.a
- 将a格式的静态库解压为o文件,可以解压出所有的.o文件
ar -x lib.a
- 将o文件合并为a文件
libtool -static -o lib.a *.o
静态库中的注意事项
- 静态库所依赖的dylib或者framework,最终使用静态库的程序也需要引用
- 静态库是二进制代码,区分处理器类型的,可以使用lipo -create –output创建支持多处理器的静态库
- 静态库中使用的开源代码与引用静态库的工程使用的开源代码相同时,将会出现冲突.
- 解决1: 静态库不打包相关的开源代码,在静态库使用说明文档中列出所依赖的开源库及其版本
- 解决2: 静态库在打包开源代码时,修改开源代码的类的命名加上三个字母以上的前缀(不建议)
- 对于已经打包好的静态库,可以利用lipo命令解包其中一个静态库,然后把发生冲突的.o文件删除,然后lipo命令重新打包,然后对每种处理器框架的.a文件重复操作,最后lipo重新合并静态库,替换原来的静态库
- a.静态库中的类的命名与引用静态库的工程的类的命名相同时将会被视为重复定义(OC没有命名空间导致)
- 解决:静态库中的类命名加上三个字母以上的前缀
- 静态库中使用Category,用Category添加的方法即使名字重复 了也不会报错,但是会存在其中一个方法覆盖了另一个
- 解决: 建议给category添加的方法添加前缀
- 全局变量重名是必然编译失败的
- 解决:给静态库中用到的全局变量加前缀
- 静态库中有可能会使用debug时打开log,release关闭log,发布到Appstore的应用不应该把调试的log打印出来,因此,在打包静态库时,把Build Configuration选择为Release一般可以把多数开源库的log都去掉了,如果静态库使用者希望能在开发时看到log,这时就看不到了
- 解决:分开打debug和release的包