若是咱们有些功能要给别人用,可是又不想公开代码实现,好比高德地图、第三方登陆分享等等,这时候咱们就要打包成库了。库分静态库和动态库两种:php
静态库:以
.a
和.framework
为文件后缀名。
动态库:以.tbd
(以前叫.dylib
) 和.framework
为文件后缀名。架构
静态库与动态库的区别框架
静态库:连接时会被完整的复制到可执行文件中,被屡次使用就有多份拷贝。
动态库:连接时不复制,程序运行时由系统动态加载到内存,系统只加载一次,多个程序共用(如系统的UIKit.framework
等),节省内存。iphone可是苹果不让使用本身的动态库,不然审核就没法经过。ui
咱们先来看一下iOS设备有哪些架构,由于下面要用到:.net
模拟器:
iPhone4s-iPnone5:i386
iPhone5s-iPhone7 Plus:x86_64调试真机:
iPhone3gs-iPhone4s:armv7
iPhone5-iPhone5c:armv7s
iPhone5s-iPhone7 Plus:arm64code支持armv7的静态库能够在armv7s上正常运行。blog
.a静态库的制做ip
一、先建立一个新的Xcode工程Test,须要选择下面这个模板:
建立完成后是这个样子的:
二、咱们把默认生成的Test.h和Test.m删掉,从新建立一个类PrintString,在这个类里面添加一个单纯打印字符串的简单方法:
三、选择添加公开头文件
为了让使用者知道有哪些方法能够用,咱们须要公开头文件,这里咱们公开PrintString.h:
四、修改配置
咱们须要把Build Active Architecture Only
修改成NO,不然生成的静态库就只支持当前选择设备的架构。
五、而后编译
咱们分别选择Generic iOS Device
和任意一个模拟器
各编译一次,编译完后,咱们会看到工程中Products文件夹下的libTest.a由红色变成了黑色,而后show in finder
,看看生成的文件:
咱们看到它为真机和模拟器都生成了.a静态库。里面都包含咱们选择公开的头文件。
咱们来看看静态库支持的框架:命令为lipo -info 静态库名字
咱们看到,Debug-iphoneos
里面的静态库支持的架构有armv7
和arm64
因此它只能用于真机,在模拟器上会报错。Debug-iphonesimulator
里面的静态库支持的架构有i386
和x86_64
,因此它只能用于模拟器,在真机上会报错。
若是想要让模拟器和真机通用一个静态库,咱们可使用终端命令来实现。命令格式:lipo -create 第一个.a文件的绝对路径 第二个.a文件的绝对路径 -output 最终的.a文件路径
:
咱们看到生成了一个新的libTest.a
文件。这个静态库就支持全部模拟器和全部真机了。而后咱们建立一个文件夹,把.a和头文件都放进去,咱们最终须要使用的就是这个文件夹:
注意:为了开发方便,咱们可使用生成的通用静态库,可是最终上线的使用咱们能够只导入真机的,这样工程的体积也会小一些。
使用生成的.a静态库
新建一个工程,将上面的通用静态库拖进去,导入头文件,就可使用里面的方法了。通过试验,咱们生成的静态库在真机上和模拟器上都能成功打印字符串:
.frameworke静态库的制做
一、先建立一个新的Xcode工程LibTest,须要选择下面这个模板:
建立完成后是这个样子的:
建立完成后咱们能够看到,工程自己自带一个LibTest.h
文件和一个Info.plist
文件。
二、咱们建立一个类PrintString
,添加一个单纯打印字符串的简单方法:
三、选择添加公开头文件
为了让使用者知道有哪些方法能够用,咱们须要公开头文件,咱们须要在 而且将Target->Build Phases->Headers
中的Projec
t中要暴露的头文件拖拽到Pulic
里面,这里咱们公开PrintString.h
:
注意:暴露出来的头文件中import的其余类也得添加到public中暴露出来。若是不想将import的类暴露出来,那么在头文件中用@class 而后在对应的.m文件中再import。
四、设置支持全部架构(和.a制做同样)
五、修改生成的Mach-O
格式,由于动态库也能够是以framework形式存在,因此须要设置,不然默认打出来的是动态库。将target->BuildSetting->Mach-o Type
设为Static Library
(默认为Dynamic Library
):
六、编译
咱们分别选择Generic iOS Device
和任意一个模拟器
各编译一次,编译完后,咱们会看到工程中Products文件夹下的LibTest.framework由红色变成了黑色,而后show in finder
,看看生成的文件:
咱们看到它为真机和模拟器都生成了LibTest.framework静态库。
咱们来查看静态库支持的框架:与上面不一样,命令为lipo -info framework下的二进制文件名字
若是想要让模拟器和真机通用一个静态库,咱们可使用终端命令来实现。合并的命令与上面不一样的是:framework静态库合并的不是framework,而是framework下的二进制文件,命令为:
lipo -create 第一个framework下二进制文件的绝对路径 第二个framework下二进制文件的绝对路径 -output 最终的二进制文件路径
:
而后将任何一个framework中的二进制文件替换成合并后的二进制文件,而后把framework添加到要使用的项目中便可使用。
使用生成的.framework静态库
新建一个工程,将静态库拖进去,导入头文件,就可使用里面的方法了。通过试验,咱们生成的静态库在真机上和模拟器上都能成功打印字符串:
注意:
若是静态库中有category类,则在使用静态库的项目配置中Other Linker Flags
须要添加参数-ObjC
或者-all_load
。
若是建立的framework类中使用了.tbd
,则须要在实际项目中导入.tbd
动态库。
运行调试静态库
若是你是开发静态库的人,你会发现上面的方法只是制做静态库,并无办法运行看效果和调试bug,这时候咱们能够这样:
一、新建一个专门用来开发静态库的正常工程Test:
二、添加一个静态库的target
咱们看到它生成了几样东西:
一个framework的target:在这里面修改静态库的配置们,例如支持的架构、要暴露的头文件们和Mach-O的配置。
一个LibTest文件夹:静态库里面的类们都放在这里面。
product文件夹下面的LibTest.framework:在这里show in finder找到编译后生成的静态库。
三、开发调试代码
咱们看到程序能够正常运行,并能够在动态库里面蹲点运行。方便咱们调试。
四、确保代码没问题后,选择对应的target编译生成。
五、后面的过程就与上面同样了
原文:http://bbs.520it.com/forum.php?mod=viewthread&tid=2884&page=&extra=#pid31800