写点什么

Swift 和 Objective-C 混编在有赞移动的实践

2020 年 3 月 01 日

Swift和Objective-C混编在有赞移动的实践

一、概述

随着 Xcode 11、Swift 5.1 的正式发布,Swift 目前已经实现了 ABI 稳定及模块稳定,语法及实现也比以往更加成熟稳定,所以我们在微商城和零售等业务线中尝试使用 Swift 开发部分业务,并在二方库中进行混编开发,在此我们将一些混编经验分享出来。


二、现状

同一工程内的混编,通常来讲有两种方式:


1、在宿主工程利用桥接文件(Bridging-Header.h)进行混编


  • Swift 访问 Objective-C

  • 只需要在桥接文件中(Bridging-Header.h)中导入需要暴露给 Swift 模块的 Objective-C 类,即可在 Swift 中访问相应 Objective-C 的类和方法

  • Objective-C 访问 Swift

  • 在 Objective-C 类中导入 ProductName-Swift.h,即可访问 Swift 中暴露给 Objective-C 的类和方法


2、利用 cocoapods 包管理工具,进行二/三方库混编


  • Swift 访问 Objective-C

  • 用 Swift Module 系统,需要用到的 Objective-C 类用 import xxx 进行引用,即可在 Swift 中访问相应的 Objective-C 的类和方法

  • Objective-C 访问 Swift

  • 在 Objective-C 类中导入 ProductName-Swift.h,即可访问 Swift 中暴露给 Objective-C 的类和方法


由于我们目前的业务比如商品模块、消息模块、资产模块等都是利用 cocoapods 进行模块化管理,制作成了二方库,供微商城、零售、精选等业务线使用,不建议在宿主工程直接使用 Swift 文件进行业务开发,业务代码应该放到相应的业务模块中去,因此我们将 Swift 代码放入二/三方库中,进行混编。


三、Module 系统

3.1 LLVM Module 系统

讲到混编方案,就不得不提,苹果在 2012 年 11 月提出 LLVM 的 Module 系统,简单讲就是用树形的结构化的描述来取代以往 #include ,例如传统的 #include 现在变成 importstd.io


这样做的主要意义是:


  • 语义上完整描述了一个框架的作用

  • 提高编译时的可扩展性,同一模块只需编译或导入一次,避免了头文件的多次引用、解析

  • 减少碎片化,每个模块只处理一次,环境的变化不会导致不一致


3.2 modulemap 文件

modulemap 文件就是对一个框架,一个库的所有文件的结构化描述。默认文件名是 module.modulemap 关于 LLVM module 系统更加详细的内容,可以参考 Clang 官方文档


3.3 Swift Module

苹果为 Swift 设计了 SwiftModule。SwiftModule 可以将 Swift 解析后生成对应的 modulemap 和 umbrella.h 文件,SwiftModule 增加对编译器版本的依赖,编译产物与编译器 和 Swift 版本有关。如果想要实现 Swift 和 Objective-C 的互相访问,需要 Objective-C 库,以及对应的 umbrella.h 和 modulemap 支持。其中动态库 framework 是 Xcode 支持配置并生成 header,静态库 .a 需要自己编写对应的 umbrella.h 和 modulemap。即库之间无论何种语言实现,均需要封装为 LLVM Module 来相互访问。


LLVM Module 作为苹果公司提出的特性,已经被 Swift 完全采用,在其基础上建立自己的模块系统,当我们结合 Cocoapods 的 use_modular_headers! 配置将三方库构建成静态库,或者 use_frameworks! 配置将三方库构建成动态库时,在编译产物中都会生成一个 modulemap 和 module umbrella.h 文件



可以在 Swift 文件这样引用该模块



3.4 use_ modular_ headers!

该特性是 Cocoapods 1.5.0 引入的配置,目的是为了满足 Xcode 9 以后支持的 Swift Static Libraries ,将 Swift Pods 构建成为静态库


  • 如果你的 Swift Pod 依赖于 Objective-C,那么你需要为这个 Objective-C 库启用 modular_headers

  • 对于 pod 开发者可以在 podtargetxcconfig 内添加 ’DEFINES_MODULE’=>‘YES’,对于使用者在 podfile 内添加 use_modular _headers!

  • 在 podspec 中通过 modular_headers => true 配置特定的 pod


可以参考 Cocoapods 官方文档


四、微商城架构调整

基于上面这些背景,微商城结合团队规模和实践,计划使用壳工程和模块同 git 仓库的 Cocoapods development pod 来替代现有的子项目方式封装模块,模块间依赖基于 podspec 和 podfile 中的配置进行管理。并且为了不中断团队工作和持续交付,实行 Long Term Evolution 长期演进的策略。有关 development pod 可以参考 Cocoapods 官方文档。


微商城项目初期:


所有模块均依赖 common 模块,同时所有模块也依赖了 Cocoapods 的二/三方库;在新架构中,common 被封装为 development pod, 并在 podspec 中声明依赖。


调整后,原有的子项目通过头文件暴露的方式仍旧可以访问和依赖,模块间的 Router 和 BeeHive/Bifrost 模块管理也都支持,即该过程对于需求开发团队是无痛的。


最终所有的 development pod 通过 Podfile 集成进壳工程,同时 Podfile 中增加 use _\modular_headers! ,要求 Cocoapods 使用静态库集成并生成对应 modulemap 等 support file。我们在周会上和大家同步了如何将原有的 Xcode 子项目模块迁移到 development pod ,简言之分为三个部分,声明源码,声明资源文件,声明依赖和其他配置,具体 podspec 文档可以参考 Cocoapods 官方文档。


最终整体架构如下所示:



在上述版本交付并合并到 master 后,经过完整测试,大家的开发体验没有改变。之后将业务模块也拆分为 development pod ,单个业务模块直接依赖 common pod。在迁移过程中,可以先依赖 common 以实现对二/三方库的依赖。随业务迭代,单业务 development pod 也逐渐理清自身真实的依赖,最终可以把自己的依赖写入 podspec。


五、二方库混编

关于二方库混编方案,我们整个有赞移动业务都是用 Cocoapods 来管理二/三方库,声明 use _modular_headers! 将 Swift pods 构建成静态库,目前已经在消息业务模块中已经实践成功,在线上的状况稳定。在此总结了一些混编方案所能遇到的问题。


5.1 Framework targets 不支持 Bridging-Header

通常来讲混编的时候需要在工程中创建 Swift 文件时候,Xcode 会问询是否创建 Bridging-Header 文件,点击是,系统会帮你创建一个 Bridging-Header,你可以将需要引用的 Objective-C 模块的头文件放在里面,然后你可以在 Swift 模块用 Objective-C 的类。但是编译器是不允许在 Framework 中创建 Bridging-header,因此在二/三方库中,我们不能使用桥接文件的方式进行混编 Objective-C 代码的引用,需要用 Swift Module 进行模块间的引用。


5.2 模块引用

引用其他 Objective-C 二方库需要增加命名空间(Namespace),否则会报错找不到文件


Swift 的命名空间是以模块划分的,一个模块表示一个命名空间。开发时,默认添加到主 target 的内容是同处于同一个命名空间的;如果用 Cocoapods 导入的第三方库,是以一个单独的 target 存在,不会存在命名冲突。但如果以源码的方式导入工程,很可能发生命名冲突,所以为了安全起见,第三方库都会使用命名空间这种方式来防止冲突。


5.3 C++ 混编

Objective-C 是 C++ 的超集,就如同 Objective-C 是 C 的超集,在 OS X 上同时被 GCC 和 Clang 支持编译,.mm 是 Objective-C++ 的默认后缀名,Xcode 的编译器可以识别。在.mm 文件中,Objective-C 代码和 C++ 代码都可以正常编译运行。在消息业务模块中中引用了 WCDB 这个 Objective-C++ 的库,因此在引用的时候要将引用到的 WCDB.h 头文件中的类文件的 .h 改成 .mm。


5.4 链接错误

我们将上述工作做完后引入到宿主工程中,进行编译的时候会出现链接错误,不要担心,那是因为宿主工程中缺少 Swift 的某些系统库,在宿主工程中建立一个 Swift 文件方可解决。


5.5 Swift 调用 Objective-C

将 Swift 模块文件中,用 import xxx 的形式进行模块的引用,包括 Objective-C 的二/三方库


5.6 Objective-C 调用 Swift

  • Swift 类中将需要暴露给 Objective-C 模块引用的类,用 public 申明

  • Swift 类中需要暴露给 Objective-C 的方法要用关键字 @objc

  • 在 Objective-C 类中引用 ProductName-Swift.h 头文件即可引用暴露给 Objective-C 的 Swift 的类和方法


5.7 pod spec lint 验证和发布

在 pod spec lint 验证和 pod repo push 发布命令中增加 --use–modular-headers 关键字,否则验证发布不通过


以上是在二方库混编中遇到的一些问题,以供大家参考和探讨。


六、优势

  • Swift 中二进制库的数量逐年攀升,直到 iOS13 已经有 141 个,Foundation 中的许多系统类已经由 Swift 库实现

  • ABI 稳定,(iOS12.2 系统以上)不增大包体积

  • Cocoapods 声明 use_modular_headers! 构建 Swift 静态库,不影响启动速度


七、总结

目前微商城项目已经进行了混编项目开发,比如学习中心模块是一个纯 Swift 的二方库,而消息业务模块则是一个 Swift 和 Objective-C 混编的二方库,我们后面会进行越来越多的模块开发用混编的这种形式,新的模块采用 Swift 代码,老的业务还是 Objective-C 不动这种方案。随着 Swift 越来越主流,很多大厂的 App 都用该语言进行开发,但是不能一蹴而就全部将 Objective-C 转成 Swift,而是有很长一段时间都是混编的形式存在,希望该篇文章能够对想进行混编方案的开发者提供一定的参考。


参考文献:


  • Swift 官方文档:


https://swift.org/blog/swift-5-released/


  • Clang 官方文档:


https://clang.llvm.org/dObjective-Cs/Modules.html


  • CocoaPods 官方文档:


https://guides.cocoapods.org/making/making-a-cocoapod.html


本文转载自公众号有赞 coder(ID:youzan_coder)。


原文链接


https://mp.weixin.qq.com/s?__biz=MzAxOTY5MDMxNA==&mid=2455760552&idx=1&sn=0e8fcd64a9e6ca663de8c5ac427b7ed4&chksm=8c68688dbb1fe19bb64cb5a3f53bd3311d7b7c5825d9e988a7daeb3fa6daae02718277fcb19f&scene=27#wechat_redirect


2020 年 3 月 01 日 10:003599

评论

发布
暂无评论
发现更多内容

【华为云技术分享】DLI跨源|当DLI遇见MongoDB

华为云开发者社区

数据库 mongodb dlib 数据集 华为云

IDEA 插件找不到?看这里!那就自己敲一个!

程序员小航

json IDEA 开发工具 idea插件 IntelliJ IDEA

【写作群星榜】7.24~7.31 写作平台优秀作者 & 文章排名

InfoQ写作平台官方

写作平台 排行榜

骚操作!用 CPU 烤肉,这位程序员做到了!

程序员生活志

程序员 gpu 程序人生

秒懂云通信:如何用阿里云语音通知服务(小白指南)

阿里云Edge Plus

语音

助力银行核心下移,监控分布式数据库,融天鹰眼轻松应对五大挑战

DT极客

一年多远程工作经验,说说真实的感受

盛安德软件

JVM系列之:通过一个例子分析JIT的汇编代码

程序那些事

Java JVM JIT 汇编

Spring系列第2篇:控制反转(IoC)与依赖注入(DI),晦涩难懂么?

老大哥

六字说出微服务的本质

看山

架构 微服务

Android Development最佳实践

teoking

创业公司技术体系建设-CI/CD

星际行者

CI/CD

LeetCode题解:189. 旋转数组,使用新数组Copy,JavaScript,详细注释

Lee Chen

LeetCode 前端进阶训练营

并发编程基础原理

刚刚🏂

机器学习基石第二节 学习笔记

半亩房顶

Machine Learning

实现扫码登陆的最简单方案与原理

Java小咖秀

Java 解决方案 经验总结

架构师技术领导力成长之路

IT民工大叔

领导力 架构师

新技术(区块链)--让游戏行业走的更远

CECBC区块链专委会

区块链技术 防篡改不可逆

Springboot拦截器使用及其底层源码剖析

南方有乔木兮

机器学习基石第一节 学习笔记

半亩房顶

Machine Learning

Twitter高性能分布式日志系统架构解析

俊俊哥

pulsar bookKeeper 分布式文件存储

海华大赛第一名团队聊比赛经验和心得:AI在垃圾分类中的应用

华为云开发者社区

AI 算法 数据分析 垃圾回收机制 华为云

java高并发系列 - 第11天:线程中断的几种方式

简爱W

腾讯安全领御区块链与张裕集团达成战略合作,打造高端葡萄酒区块链溯源平台

CECBC区块链专委会

产品溯源 无法篡改

Java异步之《我call(),Future在哪里》

BUZHIDAO

继oneAPI之后,英特尔为异构计算跨架构编程再添“利器” 发布全新机器编程系统

最新动态

webRTC框架下的视频主动丢帧

fumingwang

音视频 WebRTC

实践录丨如何在鲲鹏服务器OpenEuler操作系统中快速部署OpenGauss数据库

华为云开发者社区

数据库 鲲鹏920 操作系统 服务器 opengauss

(政务上链)新数据孤岛、安全风险等问题待解

CECBC区块链专委会

工作效率 公开透明 新技术

信创舆情一线--抖音、微信读书被判侵害用户个人信息权益

统小信uos

ARTS打卡Week 09

teoking

微服务架构下如何保证事务的一致性

微服务架构下如何保证事务的一致性

Swift和Objective-C混编在有赞移动的实践-InfoQ