看到pod上各类形形色色的三方库,本身确定也会想要建立一个分享一下。作组件化时也会想要将项目拆分红一个一个的私有库。而使用pod管理三方库或私有库最重要的就是要学会如何写podsepc文件。html
如下均以我本身写的一个私有库ZCPKit去介绍。ios
- 如何建立podspec文件
- 如何编写podspec文件
- 如何实现目录分层
- 如何校验podspec文件
- 别人写的例子
- 补充 校验时出现的一些问题
建立podspec文件只须要一行命令,在你本身的三方库的根目录下输入下面的命令:c++
pod spec create XXXKit
例如我本身的ZCPKitgit
而后在项目目录下将会生成一个podspec文件github
咱们先来打开ZCPKit.podspec文件看看里面的内容。基本都是针对ZCPKit的描述和介绍。sql
一些经常使用的信息介绍:segmentfault
name:框架名
version:当前版本(注意,是当前版本,假如你后续更新了新版本,须要修改此处)
summary:简要描述,在pod search ZCPKit的时候会显示该信息。
description:详细描述
homepage:页面连接
license:开源协议
author:做者
source:源码git地址
platform:支持最低ios版本
source_files:源文件(能够包含.h和.m)
public_header_files:头文件(.h文件)
resources:资源文件(配置的文件都会被放到mainBundle中)
resource_bundles:资源文件(配置的文件会放到你本身指定的bundle中)
frameworks:依赖的系统框架
vendored_frameworks:依赖的非系统框架
libraries:依赖的系统库
vendored_libraries:依赖的非系统的静态库
dependency:依赖的三方库
上面列举的信息大部分来自pod默认生成的podspec文件中的给的,其中须要重点注意标识黑体的这几个信息。xcode
你在三方库中看到的那些文件都是经过这些配置来肯定的。当pod install引入三方库时,只会引入你在podspec中配置的那些文件。框架
下面我会举例说明该如何写podsepc中的每项配置。ide
目录结构以下
ZCPKit ┗━━━━Classes ┗━━━━ZCPKit.h ┗━━━━ZCPKit.m ┗━━━━Frameworks ┗━━━━MyFramework.framework ┗━━━━Libraries ┗━━━━libZCPKit.a ┗━━━━Resources ┗━━━━MyRes.bundle
如图所示的目录结构,每一个配置项的写法以下:
source_files:配置三方库的源文件(.h或.m文件)
写法: source_files = 'Classes/ZCPKit.{h,m}' // 直接指定文件名 或: source_files = 'Classes/*.{h,m}' // Classes文件夹下的全部匹配文件 source_files = 'Classes/**/*.{h,m}' // Classes全部路径下的全部匹配文件public_header_files:配置公有的头文件(.h文件)
写法: source_files = 'Classes/ZCPKit.h' // 直接指定文件名 或: source_files = 'Classes/*.h' // Classes文件夹下的全部匹配文件 source_files = 'Classes/**/*.h' // Classes全部路径下的全部匹配文件vendored_frameworks:配置须要引用的非系统框架(**/*的写法后面再也不赘述)
写法: vendored_frameworks = 'Frameworks/MyFramework.framework'frameworks:配置依赖的系统框架
写法: frameworks = 'AVFoundation', 'CoreGraphics', 'Security', 'SystemConfiguration'vendored_libraries:配置须要引用的非系统静态库(要注意,这里的.a静态库名字必需要带lib前缀,若是引用的静态库名字没lib前缀会致使编译报错,只须要重命名加上便可)
写法: vendored_libraries = 'Frameworks/libZCPKit.a'libraries:配置依赖的系统库(要注意,这里的写法须要忽略lib前缀)
写法: libraries = 'c++', 'sqlite3', 'stdc++.6.0.9', 'z'resources:配置资源文件(.bundle,.png,.txt等资源文件,这些资源文件会被放到mainBundle中,要注意避免发生命名重复的问题)
写法: resources = 'Resources/MyRes.bundle'resource_bundles:配置指定bundle的资源文件(能够解决resources致使的命名冲突问题)
写法: resource_bundles = { 'ZCPKitBundle' => ['Resources/MyRes.bundle'], }dependency:依赖的三方库,pod库或者能够是自身的subspec
写法: dependency 'AFNetworking', '~>3.1.0' // pod三方库 dependency 'Util' // 自身的subspec 或 dependency 'AFNetworking'
附上官网连接,能够去看一看:specs-and-specs-repo
关于更详细的podspec文件写法,能够看这篇文章:Podspec语法参考 v1.2.0.beta.1
关于资源文件更详细的写法,能够看这篇文章:给 Pod 添加资源文件
使用subspec来实现目录分层。
目录分层的好处:
- 目录分层,结构清晰;
- 使用pod引入一个三方库时,能够只引入一个subspec而不用将整个三方库引入。
例如AFNetworking:
下面举例说明,如图所示的目录结构:
ZCPKit ┗━━━━Classes ┗━━━━ZCPKit.h ┗━━━━ZCPRouter ┗━━━━ZCPRouter.h ┗━━━━ZCPRouter.m ┗━━━━ZCPUtil ┗━━━━ZCPUtil.h ┗━━━━ZCPUtil.m
写法以下:
效果:
若是想有多层的目录结构还能够继续嵌套下去。
有几个须要注意的地方:
1.层级不能出现循环依赖。
- 好比类ZCPUtil.h中 #import "ZCPRouter.h",同时ZCPRouter.h中 #import "ZCPUtil.h"。这样当写podspec时就须要在Util层级中写dependency 'ZCPKit/Router',在Router层级中写dependency 'ZCPKit/Util'。如此便存在Router与Util层级之间的循环依赖。出现循环依赖时,三方库是没法成功提交到repo上的,会报依赖错误。解决办法是,层级间要尽可能解耦。
2.分层的层级不要太多,层级不要太深。
- 为了不出现上述的循环依赖错误。
3.source_files使用的是真实的物理路径,而dependency依赖其余层级时使用的是层级路径,不是真实的物理路径
- 例如:ZCPUtil.h文件的真实路径是:ZCPKit/Framework/Util,而Util层级是属于ZCPKit层级下的一个子subspec,因此当写Router层级依赖Util层级时要写:dependency 'ZCPKit/Util'而不是dependency 'ZCPKit/Framework/Util'
在podspec写好以后咱们须要验证一下编写的内容是否有误。能够在命令行的三方库当前路径下使用下面的命令:
pod lib lint (从本地验证你的pod可否经过验证) pod spec lint (从本地和远程验证你的pod可否经过验证) pod lib lint --verbose (加--verbose能够显示详细的检测过程,出错时会显示详细的错误信息) pod lib lint --allow-warnings (容许警告,用来解决因为代码中存在警告致使不能经过校验的问题) pod lib lint --help (查看全部可选参数,可选参数能够加多个)
咱们目前只是在本地写的三方库,因此只须要使用pod lib lint验证便可(这个过程会较为费时,须要稍微等待)。
我本身加了个循环依赖的错误而后再从新检测一下试试,能够看到检测到了错误(发现有错误的时候,建议用加--verbose的命令看详细错误信息):
须要注意的:
在检测的时候会很常常会出现各类乱七八糟的错误,在文件多的状况下不太容易一次经过,而且即便你的代码在Xcode中编译经过了,可是也仍是会出现不少错误。因此你须要尽可能去注意下面的几个问题:
1.当代码中存在警告时要记得使用--allow-warnings参数。
2.若是引入的文件或分好的层级须要依赖系统库或三方库,要记得使用上面介绍的libararies,frameworks等属性。
3.若是你的三方库是在一个xcode项目中编写调试,那么尽可能不要用pch。
- 由于不少类之间的小依赖会被pch给遮掩过去,这会致使在校验时出现大量错误。
4.若是想要分层,那么尽可能解耦使模块可以单独存在。
- 缘由是当分层较多层级较深时,极可能产生循环依赖的问题。
若是想要了解一些实实在在的例子,你能够把别人的三方库下载下来,podspec文件通常会跟其放在一块儿,这样就能参考别人的podspec文件是怎么写的。好比AFNetworking:
1>用pod搜索AFNetworking
2>进入AFNetworking的源码地址
3>下载并查看
1.不支持i386编译的库没法经过pod lib lint的问题()
解决pod lib lint/repo push不支持i386编译&只能真机运行的库
https://github.com/CocoaPods/...
2.在A私有库中引入了B私有库,致使pod lib lint不经过的问题
在调用命令时须要添加B私有库所在的repo源地址。
pod lib lint <文件名>.podspec --sources='<私有库所在repo的source>,https://github.com/CocoaPods/Specs.git'
3.报错 ··· error: include of non-modular header inside framework module ··· [-Werror,-Wnon-modular-include-in-framework-module]
解决办法:在pod lib lint 或者 pod spec lint 以及 pod repo push ....时候加上 --use-libraries
Cocoapods整理(一)——安装Cocoapods
Cocoapods整理(二)——使用Cocoapods
Cocoapods整理(四)——搭建Cocoapods私有库环境
本篇文章的私有库demo:ZCPKit
给 Pod 添加资源文件
解决pod lib lint/repo push不支持i386编译&只能真机运行的库
Podspec语法参考 v1.2.0.beta.1