2016/4/7
简介
什么是库?
库是程序代码的集合,是共享程序代码的一种方式
根据源代码的公开情况,库可以分为 2 种类型
开源库
公开源代码,能看到具体实现
比如 SDWebImage 、 AFNetworking
闭源库
不公开源代码,是经过编译后的二进制文件,看不到具体实现
主要分为:静态库、动态库
静态库和动态库
静态库和动态库的存在形式
静态库: .a 和 .framework
动态库: .dylib 和 .framework
静态库和动态库在使用上的区别
静态库:链接时,静态库会被完整地复制到可执行文件中, 被多次使用就有多份冗余拷贝 (左图所示)
动态库:链接时不复制,程序运行时由系统动态加载到内存,供程序调用,系统只加载一次,多个程序共用,节省内存 (右图所示)
需要注意的是:
项目中如果使用了自制的动态库,不能被上传到 AppStore
制作 .a
新建项目-> 选择 “Cocoa Touch Static Library”
添加库需要包含的源代码
选择需要暴露出来的 .h 文件, .m 文件会自动编译到 .a 文件中
选择真机设备,然后 Command+B 编译, libFYSView.a 文件从红色变为黑色,这个地方要详细说明一下操作步骤。目前很多教程创建的.a静态库是不支持i386等平台的。(支持平台类型的命令在下面会给出)
首先要点击工程 选择Edit Scheme 步骤如下图
右击 “Show In Finder” ,查看制作好的 .a 文件
-
Release-iphoneos 文件夹里面的东西是用在真机上的
-
Release-iphonesimulator 文件夹里面的东西是用在模拟器上的
-
如果 Scheme 是 Debug 模式,生成的文件夹就以 Debug 开头
如果想让一个 .a 文件能同时用在真机和模拟器上,需要进行合并
在终端输入指令
找到对应的文件,先cd这个文件夹,
cd /Users/dianshiuser/Library/Developer/Xcode/DerivedData/FYSFlashLabel-heypxodtjyktblbfxqauguwfsnsk/Build/Products/
然后执行如下操作:
lipo -create Release-iphoneos/libFYSFlashLabel.a Release-iphonesimulator/libFYSFlashLabel.a -output libFlashLabelSDK.a
.a 文件的体积(一般情况下)
-
真机用的 .a > 模拟器用的 .a
-
所合成 .a == 真机用的 .a + 模拟器用的 .a
通过 lipo –info libFlashLabelSDK.a
可以查看 .a 的类型(模拟器还是真机)
我这个是合并后的静态库文件,所以支持i386、x86_64、arm64、armv7类型
armv7是支持比iphone5旧的那些设备,armv7s是iphone5或ipad4或者以后的新设备,i386是模拟器的。arm64是iPhone5s
使用 .a
如何使用 .a
直接将 .a 、 .h 、资源文件拖拽到其他项目中即可
我把FYS_Header和FYSView合并在SDK的.h文件,方便调用。因为FYS_Header需要在多个地方应用,之前封装的.a文件不能正常访问所以就这样做了。
在使用自制的静态库时,如果遇到error:unrecognized selector sent to class这样的错误,查看Other Linker Flags,有没有设置。Other Linker Flags设置的三个值:-all_load就是会加载静态库文件中的所有成员,-ObjC就是会加载静态库文件中实现一个类或者分类的所有成员,-force_load(包的路径)就是会加载指定路径的静态库文件中的所有成员。所以对于使用runtime时候的反射调用的方法应该使用这三个中的一个进行link,以保证所有的类都可以加载到内存中供程序动态调用
出错原因:应为.a文件中有category文件,所以Xcode编译就把category中的方法错误认为是系统的内部方法,但系统中没有这个方法,所以调试的时候,就找不到该方法的实现,报出错误信息:'unrecognized selector sent to class *******'
部分援引:http://www.cocoachina.com/ios/20150226/11182.html
2016/4/8更新
当应用别人的.a库,只能在单一平台上运行时,你可以把sdk的接口都实现一个空函数然后编译一个模拟器版本或是真机版本然后合并版本就好了。(蓝牙功能的开发,因为蓝牙只能在真机上调试,所以有的SDK只支持真机)
2016/4/14更新
因为之前的教程生成静调库不支持i386等平台类型,现更新一下生成的支持i386等平台类型的静态库教程。
XCode12适配
由于XCode12 模拟器静态库支持arm64架构引发的系列问题
真机模拟器库无法合并,报错:have the same architectures (arm64) and can't be in the same fat output file
XCode12之前:
编译模拟器静态库支持i386、x86_64两种静态库
编译真机静态库支持armv7、arm64两种静态库
使用lipo -create -output命令可以将两个库合并成一个支持模拟器和真机i386、x86_64、armv7、arm64四种架构的静态库。
XCode12编译的模拟器静态库也支持了arm64,导致出现真机库和模拟器库不能合并的问题。
解决方法:
lipo XXX.a -remove arm64 -output XXX.a
或者如下设置Excluded Architectures: