使用Swift开发SDK的优点是,生成的SDK对于Obj-C或是Swift调用都不需要自己去建桥接文件,因为Swift的SDK打包时默认已经自动生成供OC调用的.h文件。OC调用时直接import,sdk中的.h文件即可。而Swift调用时,import该SDK名即可。

制作SDK的整个过程也十分简单,网上的博客多的不能再多(大多基于OC)。

使用Swift开发,里面会遇到千奇百怪的坑!所以写这篇博客的目的分享下这些坑的解决经验。

1.工程组合的方式:

开发SDK,不建议直接把SDK当APP开发,不然每次SDK的导出非常麻烦。建议采用SDK工程和Demo工程组合的方式,这样利于导出也利于调试。

首先你新建一个framework的工程A开发SDK,再建立一个普通的APP工程B,然后把工程A拖到工程B中,这样就组成了一个工程。下次在开发的时候直接使用B工程开发即可。

这里需要注意的是,假如你需要在B中使用SDK,就需要把A产出的Product里的XXX.framework供B来引用,这里需要在B的demo工程中的Build Phases里设置2-3个地方去添加,比如

(1)Link Binary With Libraries

(2)embed frameworks

(3)copy bundle resources (假如该framework有图片,xib等资源,这里有个关于bundle的注意点下面等等会提到)

2.第三方的管理:

在开发SDK时,开发者应该有一个共识,那就是千万不能把三方的源码打包进自己的SDK。比如我需要用到MBProgressHUD,我把它封进SDK里,不但程序会很大,别人再次导入也会产生编译冲突。所以最好还是使用CocoaPods,在文档中告诉别人使用的三方库和版本号。

3.桥接文件:

假如我们在Swift的SDK中需要引用到OC文件,是不需要添加桥接文件的。(在普通app工程中是需要的)swift的SDK工程会自动生成一个xxxx.h的头文件,你需要做的有2步:

(1)在.h中import你使用到OC类的头文件。

(2)在SDK的工程设置中的Headers里,把.h放到public中,如下:

假如你之前的工程使用到了桥接文件,你要把它改成SDK,你需要把桥接文件干掉,然后把里面import的东西搬到xxxsdk.h中。

4.资源文件的调用:

假如你在SDK中使用到图片,Xib,Storyboard等资源,在APP中也许我们能直接调用,比如UIImage(Named:"xxx"),但是普通的APP会默认搜索工程目录,这也称为MainBundle。

但是SDK中,资源文件是以工程目录/xxxSDK.Framework/xxx.png的形式存放的,所以按直接名字取的方法,绝对是直接崩溃的。

这个坑我还没有找到完美的解决办法。

(1)最合理的方式,必然是把所有的资源文件,包括XIB、图片等全部加入一个Bundle,但是Bundle的位置也是在SDK中,同样尴尬的无法找到。解决办法是,在该资源的位置前面,加入xxxSDK.Framework/xxx.bundle找到这个Bundle,然后再去找到图片,但是Bundle这东西,太垃圾了,非常难用。

(2)一个比较简单,但不是很规范的方法,定义一个全局变量let SourceRootUrl = "xxxSDK.Framework/",然后在每次调用资源前面,加入这个目录关系,比如图片,UIImage(Named:SourceRootUrl+"xxx"),xib,storyboard同样需要。

关于这点,我觉得Xcode在之后应该会推出更好的解决办法。

5.CPU架构的支持:

这点非常坑,Xcode能debug出或者release出的架构非常单一,要么是真机架构的,要么是模拟器架构的,但实际上我们提供给开发者的SDK可能是真机和模拟器两个平台都需要的。

所以我们可以对两个库进行合并,使用lipo -create  a b -output c (具体方式自己百度搜索lipo -create),【这里要注意的是我们合并的是xxxSDK.Framework里的xxxSDK文件,而不是整个framework,合并成c后,我们再放入替换某个a或b】

但是这里有个坑,使用swift开发出的SDK,通过lipo -create合并出的Framework,OC是模拟器和真机都能调用的,但是Swift的工程缺只能调用其中1个,可能是和这种合并不是非常规范。

这一点,官方居然不提供点更直接点的方法?

然后就是,关于armv7s的架构(仅iPhone5),Xcode默认已经不支持,所以假如你其中一个第三方不支持armv7s,你的SDK也就不支持了,人家的APP也不可能支持了。

6.重名问题:

SDK的启动类和SDK名,不要命名成一样的。否则你会后悔的。

7.删除framework后残留问题

有时候也许你明明已经把SDK从工厂中挪走了,但是编译的时候还是不通过,甚至运行的时候出现未找到xxxSDK。

为什么?因为Xcode非常傻*。你把SDK挪走了,但是你之前加过target,link with libraries,bundle等还残留着之前那个目标framework,然后一运行,还是试图去加载之前那个framework,就崩溃了,真是傻啊。

8.莫名崩溃,又莫名好了

我感觉Xcode的隐形BUG已经非常的多了,尤其是结合Storyboard或是Xib后,BUG直接多一个数量级。

所以这也是很多人建议少用界面编辑器去做APP的原因,有时候就是莫名其妙的错。

然而你可能重启下电脑,关闭下Xcode,或许clean up一下,也许根本就没问题。

所以假如出现了一个长达几小时让人自信到不可能有问题的BUG,不妨重启clean up -> 重启Xcode -> 重启电脑下。

使用Swift打造动态库SDK和DemoAPP时所遇到的(Xcode7.3)的更多相关文章

  1. swift的动态库

    共享可执行文件 iOS 有沙箱机制,不能跨App间共享共态库,但Apple开放了App Extension,可以在App和Extension间共间动态库(这也许是Apple开放动态链接库的唯一原因了) ...

  2. iOS - swift 后使用打包动态库

    WWDC2014上发布的Xcode6 beta版有了不少更新,其中令我惊讶的一个是苹果在iOS上开放了动态库,在Xcode6 Beta版的更新文档中是这样描述的: Frameworks for iOS ...

  3. iOS XCode7制作.Framework动态库和.a静态库的总结

    一.开发SDK时的支持情况: OC语言制作动态库时,支持iOS8+:OC语言制作静态库,支持iOS7+. Swift语言制作动态库时,支持iOS8+;Swift不支持静态库. 对于SDK来说,支持情况 ...

  4. iOS 静态库,动态库与 Framework 浅析

    静态库与动态库的区别 首先来看什么是库,库(Library)说白了就是一段编译好的二进制代码,加上头文件就可以供别人使用. 什么时候我们会用到库呢?一种情况是某些代码需要给别人使用,但是我们不希望别人 ...

  5. ios-静态库,动态库,framework浅析(一)

    一,所谓的“库”         * 所谓的“库”          库(Library)说白了就是一段编译好的二进制代码,加上头文件就可以供别人使用.什么时候我们会用到库呢?         一种情 ...

  6. WWDC2014之iOS使用动态库 framework【转】

    from:http://www.cocoachina.com/industry/20140613/8810.html JUN 12TH, 2014 苹果的开放态度 WWDC2014上发布的Xcode6 ...

  7. Xcode 创建静态库和动态库

    1.linux中静态库和动态库区别: 库从本质上来说是一种可执行代码的二进制格式,可以被载入内存中执行.库分静态库和动态库两种. 静态库:这类库的名字一般是libxxx.a:利用静态函数库编译成的文件 ...

  8. WWDC2014之iOS使用动态库

    苹果的开放态度 WWDC2014上发布的Xcode6 beta版有了不少更新,其中令我惊讶的一个是苹果在iOS上开放了动态库,在Xcode6 Beta版的更新文档中是这样描述的: Frameworks ...

  9. iOS 使用动态库

    苹果的开放态度 WWDC2014上发布的Xcode6 beta版有了不少更新,其中令我惊讶的一个是苹果在iOS上开放了动态库,在Xcode6 Beta版的更新文档中是这样描述的: Frameworks ...

随机推荐

  1. StringBuilder用法

    //来自MSDN,不太懂英文可以使用百度翻译. //using System; //using System.Text; //public sealed class App //{ // static ...

  2. 再探banana

    在Solr图形化界面:除Hue之外的选择中列出了banana的如下一些不足,今天再次研究这些地方是否有方案可以解决. 1.sunburst图功能没法用. 2.中文有些地方会显示%2B%4C之类的一串字 ...

  3. SQLServer复制(二)--事务代理作业

    之前的一篇已经介绍了如何配置复制,介绍了发布者.分发者和订阅者以及事务日志运行的简单关系.其中提到了复制代理,我们这篇将详细介绍复制代理,它是什么?在事务复制的步骤中起到了什么作用? 代理和工作 首先 ...

  4. 廖雪峰python教程的第一个疑问

    函数的参数一节中提到: def add_end(L = []); L.append('END') return L 正常调用add_end时(也就是有参数传入时): >>> add_ ...

  5. kobject.c 添加注释

    最近结合<Linux Device Drivers>对kobject的理解,对kobject.c文件添加注释,仅供参考! 1 /**  2  *    populate_dir - pop ...

  6. ubuntu下apache重启报Could not reliably determine the server’s fully......

    在Ubuntu上安装Apache,每次重启,都会出现以下错误提示: Could not reliably determine the server’s fully qualified domain n ...

  7. 令人崩溃的@requestBody乱码一例

    这个问题真是让我心力憔悴了...在客户现场对接就是乱码,StringHttpConverter怎么配置都不行... 场景其实很简单:客户那头post一个http请求,包体是json字符串,我这头spr ...

  8. Neutron 理解(14):Neutron ML2 + Linux bridge + VxLAN 组网

    学习 Neutron 系列文章: (1)Neutron 所实现的虚拟化网络 (2)Neutron OpenvSwitch + VLAN 虚拟网络 (3)Neutron OpenvSwitch + GR ...

  9. 解决cefsharp在winform中不显示tooltipText问题(网页元素的title提示)

    1.监听网页属性改变事件 webView.PropertyChanged += webView_PropertyChanged; 2.拖一个ToolTip控件到窗体 3.在webView_Proper ...

  10. 学习大神笔记之“MyBatis学习总结(二)”

    MyBatis对表的增删改查操作         主要有两种方式:基于XML实现和基于注解实现. 完整项目结构: 工具类:MyBatisUtil-------用于获取  sqlsession pack ...