Swift App
混入OC/C/C++代碼
方法一
使用橋接文件锈玉。
方法二
使用.modulemap
-
Swift App
項目废亭,引入OC/C/C++
文件。 - 創(chuàng)建文件名必須為
module
后綴為.modulemap
的文件穿香。
- 配置工程
BuildSetting
的Swift Compiler - Search Paths
選項,值為module.modulemap
文件所在的目錄路徑或其上層目錄路徑称开。
此處可為:
${SRCROOT}/MixFrameworkTest
${SRCROOT}/MixFrameworkTest/ObjcInSwitApp
如果這里的路徑配置不正確便會報錯:error build: No such module '*'
4.配置module.modulemap
內(nèi)容
module OcInApp {
// 所引入頭文件相對于`.modulemap`的路徑; "./OcClassInApp.h"也可以
header "OcClassInApp.h"
export *
}
-
import
導(dǎo)入使用
import OcInApp
class ViewController: UIViewController {
override func viewDidLoad() {
super.viewDidLoad()
OcClassInApp.description()
}
}
混入OC Framework
如果引入的OC FrameWork
中含有Modules/module.modulemap
文件舞骆,則直接拖入Swift
工程中钥弯,使用import
引入并進行調(diào)用即可。
如果OC FrameWork
不支持modulemap
(沒有module.modulemap
文件)督禽,則有兩種方法:
方法一
建立橋接文件脆霎,按需導(dǎo)入頭文件
方法二
基于OC FrameWork/Modules/module.modulemap
路徑,創(chuàng)建module.modulemap
文件狈惫,并輸入以下內(nèi)容并保存睛蛛,之后便可使用import
引入并進行調(diào)用。
// 按需導(dǎo)入
framework module SameAsFrameWorkName { //必須與導(dǎo)入的`OC Framework`同名
header "A.h"
header "B.h"
header "b.h"
//..
export *
}
///遞歸導(dǎo)入
framework module SameAsFrameWorkName { //必須與導(dǎo)入的`OC Framework`同名
umbrella header "SameAsFrameWorkName.h" //SDK對外頭文件胧谈,包含SDK對外暴露的諸多.h文件
export *
module * { export * }
}
混入OC xcframework
與混入OC framework
操作基本一致忆肾,如果引入的OC xcframework
下不同的架構(gòu)文件下的framework
中含有Modules/module.modulemap
文件,則直接拖入Swift
工程中菱肖,使用import
引入并進行調(diào)用即可客冈。
如果OC xcframework
不支持modulemap
,也是有兩種方法:
方法一
建立橋接文件稳强,按需導(dǎo)入頭文件场仲。
方法二
對OC xcframework
下不同架構(gòu)framework
和悦,參考混入OC framework
的方法二,即可調(diào)用渠缕。
Swift Framework
當Swift Framework
鸽素,僅有Swift
類時,如果是需要SDK
外部使用的Class
或Method
亦鳞,則只需要使用public
或open
修飾馍忽。
在Swift
工程導(dǎo)入Swift Framework
后,直接在Swift
的工程中使用即可燕差。
import RadarSDK
class ViewController: UIViewController {
override func viewDidLoad() {
super.viewDidLoad()
RadarEntry.start()
}
}
注意:Swift Framework
中是不支持使用橋接文件的遭笋,不然的話會報錯:
混入OC/C/C++代碼
示例工程結(jié)構(gòu)如下:
內(nèi)部使用 OC
SDK
內(nèi)部的Swift
類,不能直接使用OC
的代碼徒探,并且不支持橋接文件坐梯。因此只能使用module.modulemap
文件
- 創(chuàng)建文件名必須為
module
后綴為.modulemap
的文件。 - 配置工程
BuildSetting
的Swift Compiler - Search Paths
選項刹帕,值為module.modulemap
文件所在的目錄路徑或其上層目錄路徑,此處可為:
${SRCROOT}/SwiftMixSDK/ObjcSources
${SRCROOT}/SwiftMixSDK
- 配置
module.modulemap
內(nèi)容
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
外部需要調(diào)用混入的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()
}
}
- 編譯最終產(chǎn)物對外暴露的頭文件
或
方法二
自定義Swift Framework
的.modulemap
文件钱贯。
- 自定義
module.modulemap
文件挫掏;此處的文件名不強制module
,但建議用module
秩命,因為編譯器最終會合并自定義的文件尉共,最終導(dǎo)出module.modulemap
文件。 - 配置
Frame Target
的build Setting
弃锐,保證Define Module
為YES
;Module Map File
為自定義.modulemap
文件的路徑袄友。
- 配置
module.modulemap
內(nèi)容
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()
}
}
- 編譯最終產(chǎn)物對外暴露的頭文件
小結(jié)
綜上可以看出,兩種方法編譯的最終產(chǎn)物存在差異:
方法一會暴露OC
頭文件的旋廷;方法二會隱藏OC
的頭文件鸠按。
多人協(xié)作開發(fā)Swift Framework
時,方法二會存在頻繁修改.modulemap
文件文件饶碘。因此實際開發(fā)中還得基于項目決定方案目尖。
混入OC Framework
當OC Framework
不支持Module
時,有兩種方式混入OC Framework
扎运。
方法一
通過使用.modulemap
文件瑟曲,實現(xiàn)對OC Framework
調(diào)用饮戳。即:OC Framework
支持(自動或者手動)module
。
方法二
參考上述 混入OC/C/C++代碼 —— 內(nèi)部使用OC 测蹲,創(chuàng)建module.modulemap
文件莹捡,配置相關(guān)Building Setting
。配置module.modulemap
內(nèi)容如下:
module ObjcFramwork {
///相對于module.modulemap的相對路徑
umbrella header "../ObjcFramwork.framework/Headers/ObjcFramwork.h"
export *
module * { export * }
}
最后在Swift
文件中import ObjcFramwork
扣甲,即可調(diào)用篮赢。
混入OC xcframework
當OC xcframework
不支持Module
時,有兩種方式混入OC xcframework
琉挖。
方法一
對OC xcframework
下不同架構(gòu)framework
启泣,參考混入OC Framework
的方法一,進行操作后即可調(diào)用示辈。
方法二
參考上述 混入OC/C/C++代碼 —— 內(nèi)部使用OC 寥茫,創(chuàng)建module.modulemap
文件,配置相關(guān)Building Setting
矾麻;
這些操作與混入OC Framework
的方法二一致纱耻,唯一有區(qū)別在于:OC xcframework
包含多種架構(gòu)的Framework
。
因此不能通過相對路徑直接引入险耀,因為不同架構(gòu)路徑下的同一個頭文件會相互覆蓋而報錯弄喘;
如何解決呢?我們可以通過新建一個ObjcXCFramwork-umbrella.h
文件(可以是其它名稱)甩牺,并在其中引入xcframework
的頭文件:
#import <MyTestSDK/MyTestSDK.h>
然后配置module.modulemap
內(nèi)容如下:
module ObjcXCFramwork {
umbrella header "ObjcXCFramwork-umbrella.h"
export *
module * { export * }
}
最后在Swift
文件中import ObjcXCFramwork
蘑志,即可調(diào)用。