自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

Swift TIP之objc 和 dynamic

移動開發(fā) iOS
雖然說 Swift 語言的初衷是希望能擺脫 Objective-C 的沉重的歷史包袱和約束,但是不可否認的是經(jīng)過了二十多年的洗禮,Cocoa 框架早就烙上了不可磨滅的 Objective-C 的印記。無數(shù)的第三方庫是用 Objective-C 寫成的,這些積累無論是誰都不能小覷。因此,在最初的版本中,Swift 不得不考慮與 Objective-C 的兼容。

雖然說 Swift 語言的初衷是希望能擺脫 Objective-C 的沉重的歷史包袱和約束,但是不可否認的是經(jīng)過了二十多年的洗禮,Cocoa 框架早就烙上了不可磨滅的 Objective-C 的印記。無數(shù)的第三方庫是用 Objective-C 寫成的,這些積累無論是誰都不能小覷。因此,在最初的版本中,Swift 不得不考慮與 Objective-C 的兼容。

Apple 采取的做法是允許我們在同一個項目中同時使用 Swift 和 Objective-C 來進行開發(fā)。其實一個項目中的 Objective-C 文件和 Swift 文件是處于兩個不同世界中的,為了讓它們能相互聯(lián)通,我們需要添加一些橋梁。

首先通過添加 {product-module-name}-Bridging-Header.h 文件,并在其中填寫想要使用的頭文件名稱,我們就可以很容易地在 Swift 中使用 Objective-C 代碼了。Xcode 為了簡化這個設定,甚至在 Swift 項目中第一次導入 Objective-C 文件時會主動彈框進行詢問是否要自動創(chuàng)建這個文件,可以說是非常方便。

但是如果想要在 Objective-C 中使用 Swift 的類型的時候,事情就復雜一些。如果是來自外部的框架,那么這個框架與 Objective-C 項目肯定不是處在同一個 target 中的,我們需要對外部的 Swift module 進行導入。這個其實和使用 Objective-C 的原來的 Framework 是一樣的,對于一個項目來說,外界框架是由 Swift 寫的還是 Objective-C 寫的,兩者并沒有太大區(qū)別。我們通過使用 2013 年新引入的 @import 來引入 module:

 

  1. @import MySwiftKit; 

之后就可以正常使用這個 Swift 寫的框架了。

如果想要在 Objective-C 里使用的是同一個項目中的 Swift 的源文件的話,可以直接導入自動生成的頭文件 {product-module-name}-Swift.h 來完成。比如項目的 target 叫做 MyApp 的話,我們就需要在 Objective-C 文件中寫

 

  1. #import "MyApp-Swift.h" 

但這只是故事的開始。Objective-C 和 Swift 在底層使用的是兩套完全不同的機制,Cocoa 中的 Objective-C 對象是基于運行時的,它從骨子里遵循了 KVC (Key-Value Coding,通過類似字典的方式存儲對象信息) 以及動態(tài)派發(fā) (Dynamic Dispatch,在運行調(diào)用時再決定實際調(diào)用的具體實現(xiàn))。而 Swift 為了追求性能,如果沒有特殊需要的話,是不會在運行時再來決定這些的。也就是說,Swift 類型的成員或者方法在編譯時就已經(jīng)決定,而運行時便不再需要經(jīng)過一次查找,而可以直接使用。

顯而易見,這帶來的問題是如果我們要使用 Objective-C 的代碼或者特性來調(diào)用純 Swift 的類型時候,我們會因為找不到所需要的這些運行時信息,而導致失敗。解決起來也很簡單,在 Swift 類型文件中,我們可以將需要暴露給 Objective-C 使用的任何地方 (包括類,屬性和方法等) 的聲明前面加上 @objc 修飾符。注意這個步驟只需要對那些不是繼承自 NSObject 的類型進行,如果你用 Swift 寫的 class 是繼承自 NSObject 的話,Swift 會默認自動為所有的非 private 的類和成員加上 @objc。這就是說,對一個 NSObject 的子類,你只需要導入相應的頭文件就可以在 Objective-C 里使用這個類了。

@objc 修飾符的另一個作用是為 Objective-C 側(cè)重新聲明方法或者變量的名字。雖然絕大部分時候自動轉(zhuǎn)換的方法名已經(jīng)足夠好用 (比如會將 Swift 中類似 init(name: String) 的方法轉(zhuǎn)換成 -initWithName:(NSString *)name 這樣),但是有時候我們還是期望 Objective-C 里使用和 Swift 中不一樣的方法名或者類的名字,比如 Swift 里這樣的一個類:

 

  1. class 我的類 { 
  2. func 打招呼(名字: String) { 
  3. println("哈嘍,\(名字)"

我的類().打招呼("小明")

Objective-C 的話是無法使用中文來進行調(diào)用的,因此我們必須使用 @objc 將其轉(zhuǎn)為 ASCII 才能在 Objective-C 里訪問:

 

  1. @objc(MyClass) 
  2. class 我的類 { 
  3. @objc(greeting:) 
  4. func 打招呼(名字: String) { 
  5. println("哈嘍,\(名字)"

這樣,我們在 Objective-C 里就能調(diào)用 [[MyClass new] greeting:@"XiaoMing"] 這樣的代碼了 (雖然比起原來一點都不好玩了)。另外,正如上面所說的以及在 Selector 一節(jié)中所提到的,即使是 NSObject 的子類,Swift 也不會在被標記為 private 的方法或成員上自動加 @objc。如果我們需要使用這些內(nèi)容的動態(tài)特性的話,我們需要手動給它們加上 @objc 修飾。

添加 @objc 修飾符并不意味著這個方法或者屬性會變成動態(tài)派發(fā),Swift 依然可能會將其優(yōu)化為靜態(tài)調(diào)用。如果你需要和 Objective-C 里動態(tài)調(diào)用時相同的運行時特性的話,你需要使用的修飾符是 dynamic。一般情況下在做 app 開發(fā)時應該用不上,但是在施展一些像動態(tài)替換方法或者運行時再決定實現(xiàn)這樣的 "黑魔法" 的時候,我們就需要用到 dynamic 修飾符了。在之后的 KVO 一節(jié)中,我們還會提到一個關(guān)于使用 dynamic 的實例。
 

責任編輯:chenqingxiang 來源: swifter.tips
相關(guān)推薦

2015-02-05 00:18:44

SwiftObjective-C

2021-07-09 19:04:55

Cache查找消息

2015-04-01 14:34:37

C#dynamicDictionary性

2015-08-25 14:25:54

objc_msgsen

2009-05-25 15:42:03

Visual StudC#

2015-07-30 10:42:38

SwiftUIButton

2015-03-16 10:17:48

objc照片框架

2014-06-05 14:12:05

SwiftUI學習iOS

2024-01-15 08:39:12

ArraysSetsTypeScript

2019-03-26 08:15:45

iOS尾調(diào)用Objective-C

2011-01-14 13:40:24

UbuntuLinux Tips

2011-12-29 15:35:39

Web

2010-09-09 12:49:58

鼠標懸停tip效果CSS

2009-10-27 11:08:25

C# 4.0dynamic

2022-03-04 09:02:01

StoryBoard工具git

2010-12-03 12:16:49

2014-09-26 09:49:48

SwiftObjective-C

2025-03-19 09:02:18

Debouncing任務讓步Swift

2023-02-08 09:01:42

Swift元素流

2009-09-25 10:38:42

Hibernate動態(tài)
點贊
收藏

51CTO技術(shù)棧公眾號