.framework是什么?

这个问题相信做iOS的都知道答案。 在我们的日常开发中,经常会用到各种已经封装好的库,比如支付宝、微信SDK等等中的库,这些库可以给我们的开发带来很大的便利。有的时候,由于工作的需要,我们需要对自己的项目进行封装,生成库,方便别人的使用。在这里就边参考好点的博客,边总结一下我们经常看到的.framework。

那什么是“库”呢?

“库”是共享程序代码的一种方式!同行总结的这句话很简单也很好的说明了它的作用!

一般的分为“静态库”和“动态库”。

“静态库”和“动态库”有什么区别?

“静态库”  链接时候完整的拷贝至可执行文件中,被多次使用就会有多次拷贝。

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

iOS里静态和动态库形式

静态库形式: .a和.framework

动态库形式: .dylib和.framework

.a与.framework有什么区别

.a是一个纯二进制文件,.framework中除了有二进制文件之外还有资源文件。

.a文件不能直接使用,至少要有.h文件配合,.framework文件可以直接使用。

.a + .h + sourceFile = .framework。

所以我们建议用.framework.

下面我们通过实际的例子自己制作一下

我们首先得创建这个FrameWork,按照下图:


我们创建了这个FrameWork,看看这个 FrameWork的结构:


我们再里面简单的添加了一个ShowNSlog的类,并且我们添加了一个 FrameWorkTest.boundle文件,看看我们在这个类里面写了什么内容:

#import "ShowNSLog.h"
@implementation ShowNSLog
+(void)showLog{

  NSLog(@"你使用了我们的静态库");
}
+(NSString *)showLogWithReturn{

  return @"zhouzhou.jpg";
}
@end

接下来对我们的这个.framework静态库进行一些简单的设置,如下图所示:

1、首先是Dead Code Stripping设置为NO,网上对此项的解释如下,大致意思是如果开启此项就会对代码中的”dead”、”unreachable”的代码过滤,不过这个开关是否关闭,似乎没有多大影响,不过为了完整还原framework中的代码,将此项关闭也未曾不可。

2、然后将Link With Standard Libraries关闭,我想可能是为了避免重复链接

3、最后将Mach-O Type设为Static Library,framework可以是动态库也可以是静态库,对于系统的framework是动态库,而用户制作的framework只能是静态库。


接下里就是设置我们有那些头文件是需要公开的,如下图设置:


还要记得把要公开的类添加到我们的FrameWorkTest.h中,比如下面是我们例子中的:

#import
//! Project version number for FrameWorkTest.
FOUNDATION_EXPORT double FrameWorkTestVersionNumber;

//! Project version string for FrameWorkTest.
FOUNDATION_EXPORT const unsigned char FrameWorkTestVersionString[];

// In this header, you should import all the public headers of your framework using statements like #import
// 导入要公开的头文件
#import

最后要做的就是打包制作我们这个FrameWork了:

command+B 按照我们下图的选择,打包出这个FrameWork:


这个时候你就会看到FrameWork项目里的Products文件多了我们的.framework文件。你Show in Finder一下就会看到下面这样的两个文件夹了,一个就是真机一个就是模拟机的:


接下来就是生成我们.framework文件的最后一步了:利用终端把模拟机和真机的文件我们合并成一份:

把上图中我们标注的FrameWorkTest文件进行下面的操作:

在终端中输入命令: lipo -create  模拟机和真机的FrameWorkTest文件路径(直接拉到终端就会显示)-output 一个输出路径 

具体的例子我们看下面我们终端中的信息:


注意:随后生成的可以看到是一个.lipo文件,这时候你需要做的就是改了它的名称(包括去掉后缀)然后去随便覆盖你的真机或者模拟机的之前我们合并时候的文件!

然后就是它的使用了

我们的.framework文件就算是制作完成了,那使用我相信大家也都知道,把它拉到我们的项目中,我们看看我们的使用情况,证明我们的是OK的:

顺便这里说一下.boundle文件的制作,新建一个文件,把它后缀名改为.boundle文件,这时候你要是直接打开这个文件的不行的,那就“显示包内容”给里面添加资源文件就可以!

我们再看我们的项目中我们拉进来的文件:


我们在我们的.boundle文件中添加了一张图片,我们再具体看看这个图片的使用以及我们这个.framework静态库是不是能成功使用:

一:导入我们的静态库:


二:具体的使用:


三:看看我们的打印日志:


上面的哪张我们使用的图片也是能够正常出现的,我们这里就不再截图发了,有兴趣的可以自己试试。

最后需要总结的:

1、在制作framework或者lib的时候,如果使用了category,则使用改FMWK的程序运行时会crash,此时需要在该工程中 other linker flags添加两个参数 -ObjC -all_load。(这点没有亲测)

2、带有图片资源的需要把图片打包成Bundle文件,和framework一起拷贝到相应的项目中。

3、公开的类中如果引用的private的类,打包以后对外会报错,找不到那个private的类,可以把那个private的.h放到(也没亲测)

4、namespace 冲突。静态库用了某第三方库,项目也用了同样的第三方库,在编译的时候就会有 duplicate symbol 错误,因为有两份同样的第三方库。解决办法就是把用到的第三方库加上自定义前缀,包括类名、delegate 协议、常量名,尤其需要注意 Category 的方法名要修改。

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对脚本之家的支持。

02-07 11:29
查看更多