Objective-C 和 Swift 混编项目的小 Tips(一)
本文主要閑聊一些 Objective-C 和 Swift 混編項(xiàng)目帶來(lái)的一些潛規(guī)則,希望能幫到對(duì)此感到疑惑的朋友。下面我們開(kāi)始進(jìn)入主題:
命名
官方 Guide 上只是簡(jiǎn)單敘述(Using Swift with Cocoa and Objective-C),即 Swift 編譯器會(huì)在我們使用 Objective-C 的 API 時(shí)自動(dòng)的將其轉(zhuǎn)成 Swift 風(fēng)格的 API(說(shuō)白了就是會(huì)對(duì)一些方法名、枚舉名等等做一些有規(guī)則的刪減,即重命名)。
單例方法命名
在 Swift 中引用 Objective-C 單例時(shí),如果單例方法包含于類(lèi)名,則會(huì)出現(xiàn)編譯錯(cuò)誤,下面我們來(lái)看幾個(gè)例子。
Example 1
@interface TXLocationManager : NSObject+ (instancetype)manager;@end復(fù)制代碼TXLoginManager 類(lèi)有一個(gè)單例方法命名為 manager,在 Swift 中引用 manager 方法時(shí),會(huì)出現(xiàn)編譯錯(cuò)誤:
說(shuō)白了,manager 方法已經(jīng)廢了。。。
Example 2
在 Example 1 的基礎(chǔ)上,我們把單例方法的命名改一改:
@interface TXLocationManager : NSObject+ (instancetype)shareInstance;@end復(fù)制代碼單例方法命名改成 shareInstance 后,編譯通過(guò)了。至此,至少問(wèn)題已經(jīng)解決了,現(xiàn)在我們?cè)俸?jiǎn)單看看是什么原因?為何 manager 方法無(wú)法引用,而 shareInstance 卻可以引用呢?
Example 3
在 Example 1 的基礎(chǔ)上,把 manager 單例方法名稱(chēng)改為 shareManager :
@interface TXLocationManager : NSObject+ (instancetype)shareManager;@end復(fù)制代碼我們可以發(fā)現(xiàn)在 Swift 中引用時(shí),shareManager 方法名被重命名為 share :
小結(jié)
至此,我們可以得出一個(gè)簡(jiǎn)單的命名潛規(guī)則:在 Swift 中引用 Objective-C 單例時(shí),如果單例方法包含于類(lèi)名,則會(huì)出現(xiàn)編譯錯(cuò)誤,準(zhǔn)確的說(shuō),應(yīng)該是如果單例方法的名稱(chēng)正好是該類(lèi)名駝峰命名的后綴,那么在 Swift 中引用該單例方法時(shí),會(huì)出現(xiàn)編譯錯(cuò)誤。
為何在 Swift 中引用 Objective-C 類(lèi)的 API 會(huì)出現(xiàn)這種問(wèn)題呢?官方 Guide 上時(shí)這樣描述的:
The Swift compiler automatically imports Objective-C code as conventional Swift code. It imports Objective-C class factory methods as Swift initializers, and Objective-C enumeration cases truncated names.
因?yàn)?Swift 編譯器在使用 Objective-C 的代碼時(shí)會(huì)自動(dòng)的將其轉(zhuǎn)成 Swift 風(fēng)格的代碼,就是會(huì)對(duì)一些方法名、枚舉名等等做一些有規(guī)則的刪減。
There may be edge cases in your code that are not automatically handled. If you need to change the name imported by Swift of an Objective-C method, enumeration case, or option set value, you can use the NS_SWIFT_NAME macro to customize how a declaration is imported.
根據(jù)官方 Guide,上述的這種 case 屬于 特殊的情況。那如何解決這種問(wèn)題呢,Swift 提供了一個(gè)宏,專(zhuān)門(mén)處理我們遇到的這種 case —— NS_SWIFT_NAME
@interface TXLocationManager : NSObject+ (instancetype)manager NS_SWIFT_NAME(shareInstance());@end復(fù)制代碼這樣,manager 該單例方法,當(dāng)我們?cè)?Swift 中引用時(shí),會(huì)被重命名為 shareInstance。
let _ = TXLocationManager.shareInstance()復(fù)制代碼普通方法命名
有時(shí)候,我們?cè)?Swift 中引用 Objective-C 中某個(gè)類(lèi)的 API 時(shí),方法名是可能會(huì)被重命名的,下面我們直接看例子。
類(lèi)方法
@interface TXLocationManager : NSObject+ (instancetype)managerWithCoordinateY:(CGFloat)y// Or // + (TXLocationManager *)managerWithCoordinateY:(CGFloat)y@end復(fù)制代碼當(dāng)該類(lèi)的類(lèi)方法返回自身類(lèi)型的實(shí)例對(duì)象時(shí),上述的方法會(huì)被重命名。應(yīng)該這樣引用:
// 方式一: let _ = TXLocationManager.init(coordinateY: 9)// 方式二: let _ = TXLocationManager(coordinateY: 9)// 錯(cuò)誤引用方式,編譯失敗 let _ = TXLocationManager.manager(withCoordinateY: 9)復(fù)制代碼通過(guò)上述實(shí)踐,我們可以發(fā)現(xiàn)類(lèi)方法中的 manager 前綴會(huì)被刪掉,而且變成了 Swift 中的 init 方法。如果該類(lèi)的類(lèi)方法不返回自身類(lèi)型的實(shí)例對(duì)象呢?
@interface TXLocationManager : NSObject+ (void)managerWithCoordinateY:(CGFloat)y;// Or // + (NSObject *)managerWithCoordinateY:(CGFloat)y; // + (CGFloat)managerWithCoordinateY:(CGFloat)y;@end復(fù)制代碼通過(guò)實(shí)踐可以發(fā)現(xiàn),在 Swift 中是可以這樣引用的:
TXLocationManager.manager(withCoordinateY: 9)復(fù)制代碼這種方式的引用同我們一般的方法引用是一致的,無(wú)異同。
實(shí)例方法
實(shí)例方法的重命名規(guī)則與類(lèi)方法有點(diǎn)相似,此處就不再詳述了,感興趣的朋友可以自己實(shí)踐一下。(當(dāng)然方法的重命名我們一般都可以通過(guò) NS_SWIFT_NAME 來(lái)指定)
參考鏈接
How to call an Objective-C singleton from Swift?
總結(jié)
以上是生活随笔為你收集整理的Objective-C 和 Swift 混编项目的小 Tips(一)的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: kotlin 中 lambda 表达式的
- 下一篇: 两串旋转