Swift 调用 OC 实用技巧
Swift App
混入OC/C/C++代码
方法一
使用桥接文件。
方法二
使用.modulemap
Swift App
项目,引入OC/C/C++
文件。- 创建 文件名必须为
module
后缀为.modulemap
的文件。
- 配置工程
BuildSetting
的Swift Compiler - Search Paths
选项,值为module.modulemap
文件所在的目录路径或其上层目录路径。
此处可为:
${SRCROOT}/MixFrameworkTest
${SRCROOT}/MixFrameworkTest/ObjcInSwitApp
如果这里的路径配置不正确便会报错:error build: No such module '*'
4.配置module.modulemap
内容
module OcInApp {
// 所引入头文件相对于`.modulemap`的路径; "./OcClassInApp.h"也可以
header "OcClassInApp.h"
export *
}
import
导入使用
import OcInApp
class ViewController: UIViewController {
override func viewDidLoad() {
super.viewDidLoad()
OcClassInApp.description()
}
}
混入OC Framework
如果引入的OC FrameWork
中含有Modules/module.modulemap
文件,则直接拖入Swift
工程中,使用import
引入并进行调用即可。
如果OC FrameWork
不支持modulemap
(没有module.modulemap
文件),则有两种方法:
方法一
建立桥接文件,按需导入头文件
方法二
基于OC FrameWork/Modules/module.modulemap
路径,创建module.modulemap
文件,并输入以下内容并保存,之后便可使用import
引入并进行调用。
// 按需导入
framework module SameAsFrameWorkName { //必须与导入的`OC Framework`同名
header "A.h"
header "B.h"
header "b.h"
//..
export *
}
///递归导入
framework module SameAsFrameWorkName { //必须与导入的`OC Framework`同名
umbrella header "SameAsFrameWorkName.h" //SDK对外头文件,包含SDK对外暴露的诸多.h文件
export *
module * { export * }
}
混入OC xcframework
与混入OC framework
操作基本一致,如果引入的OC xcframework
下不同的架构文件下的framework
中含有Modules/module.modulemap
文件,则直接拖入Swift
工程中,使用import
引入并进行调用即可。
如果OC xcframework
不支持modulemap
,也是有两种方法:
方法一
建立桥接文件,按需导入头文件。
方法二
对OC xcframework
下不同架构framework
,参考 混入OC framework
的方法二 ,即可调用。
Swift Framework
当Swift Framework
, 仅有Swift
类时 ,如果是需要SDK
外部使用的Class
或Method
,则只需要使用public
或open
修饰。
在Swift
工程导入Swift Framework
后,直接在Swift
的工程中使用即可。
import RadarSDK
class ViewController: UIViewController {
override func viewDidLoad() {
super.viewDidLoad()
RadarEntry.start()
}
}
注意: Swift Framework
中是不支持使用桥接文件的,不然的话会报错:
混入OC/C/C++代码
示例工程结构如下:
内部使用 OC
SDK
内部的Swift
类,不能直接使用OC
的代码,并且不支持桥接文件。因此只能使用module.modulemap
文件
- 创建 文件名必须为
module
后缀为.modulemap
的文件。 - 配置工程
BuildSetting
的Swift Compiler - Search Paths
选项,值为module.modulemap
文件所在的目录路径或其上层目录路径,此处可为:
${SRCROOT}/SwiftMixSDK/ObjcSources
${SRCROOT}/SwiftMixSDK
- 配置
module.modulemap
内容
module ObjcInFramwork {
header "ObjcClassA.h"
header "ObjcClassB.h"
export *
}
- 使用
import ObjcInFramwork
public class SwiftMixTest: NSObject {
public static func mixTest() {
ObjcClassA.description()
print("Swift MixIn OC")
}
}
外部使用OC
如果Swift Framework
外部需要调用混入的OC
,有两种方法:
方法一
Swift Framework
外接.h
头文件以#import <SwiftMixSDK/PublicHeader.h>
的方法对外公开需要使用的OC
头文件。
- 配置
Frame Target
的build Phases
,使得OC
头文件公开
或
- 在
Swift Framework
外接.h
文件import
需要公开的OC
头文件
- 使用
import UIKit
import SwiftMixSDK
class ViewController: UIViewController {
override func viewDidLoad() {
super.viewDidLoad()
SwiftMixTest.mixTest()
ObjcClassA.description()
ObjcClassB.description()
}
}
- 编译最终产物对外暴露的头文件
或
方法二
自定义Swift Framework
的.modulemap
文件。
- 自定义
module.modulemap
文件;此处的文件名不强制module
,但建议用module
,因为编译器最终会合并自定义的文件,最终导出module.modulemap
文件。 - 配置
Frame Target
的build Setting
,保证Define Module
为YES
;Module Map File
为自定义.modulemap
文件的路径。
- 配置
module.modulemap
内容
framework module SwiftMixSDK {
umbrella header "SwiftMixSDK.h"
export *
module * {export *}
module ObjcInFramwork {
header "/Users/*/Desktop/*/SwiftSDKExample/SwiftMixSDK/ObjcSources/ObjcClassA.h"
header "/Users/*/Desktop/*/SwiftSDKExample/SwiftMixSDK/ObjcSources/ObjcClassB.h"
export *
}
}
最终SDK
的modulemap
:
值得注意的是,此处的header
,只能使用绝对路径,否则会出错。 stackoverflow
此问题的QA
- 使用
import UIKit
import SwiftMixSDK.ObjcInFramwork
class ViewController: UIViewController {
override func viewDidLoad() {
super.viewDidLoad()
SwiftMixTest.mixTest()
ObjcClassA.description()
ObjcClassB.description()
}
}
- 编译最终产物对外暴露的头文件
小结
综上可以看出,两种方法编译的最终产物存在差异:
方法一 会暴露OC
头文件的; 方法二 会隐藏OC
的头文件。
多人协作开发Swift Framework
时, 方法二 会存在频繁修改.modulemap
文件文件。因此实际开发中还得基于项目决定方案。
混入OC Framework
当OC Framework
不支持Module
时,有两种方式混入OC Framework
。
方法一
通过使用.modulemap
文件,实现对OC Framework
调用。即:OC Framework
支持(自动或者手动)module
。
方法二
参考上述 混入OC/C/C++代码 —— 内部使用OC ,创建module.modulemap
文件,配置相关Building Setting
。配置module.modulemap
内容如下:
module ObjcFramwork {
///相对于module.modulemap的相对路径
umbrella header "../ObjcFramwork.framework/Headers/ObjcFramwork.h"
export *
module * { export * }
}
最后在Swift
文件中import ObjcFramwork
,即可调用。
混入OC xcframework
当OC xcframework
不支持Module
时,有两种方式混入OC xcframework
。
方法一
对OC xcframework
下不同架构framework
,参考混入OC Framework
的 方法一 ,进行操作后即可调用。
方法二
参考上述 混入OC/C/C++代码 —— 内部使用OC ,创建module.modulemap
文件,配置相关Building Setting
;
这些操作与混入OC Framework
的 方法二 一致,唯一有 区别 在于:OC xcframework
包含多种架构的Framework
。
因此不能通过相对路径直接引入,因为不同架构路径下的同一个头文件会相互覆盖而报错;
如何解决呢?我们可以通过新建一个ObjcXCFramwork-umbrella.h
文件(可以是其它名称),并在其中引入xcframework
的头文件:
#import <MyTestSDK/MyTestSDK.h>
然后配置module.modulemap
内容如下:
module ObjcXCFramwork {
umbrella header "ObjcXCFramwork-umbrella.h"
export *
module * { export * }
}
最后在Swift
文件中import ObjcXCFramwork
,即可调用。
本文转自Swift 调用 OC 实用技巧