Swift和Objective-C混編的注意啦

前言

Swift已推出數(shù)年,與Objective-C相比Swift的語言機(jī)制及使用簡易程度上更接地氣,大大降低了iOS入門門檻。當(dāng)然這對新入行的童鞋們來講,的確算是福音,但對于整個(gè)iOS編程從業(yè)者來講,真真是,曾幾何時(shí)“高大上”,轉(zhuǎn)瞬之間“矮矬窮”。再加上培訓(xùn)班橫行,批量批發(fā)之下,iOS再也看不到當(dāng)年的輝煌。


Swift.png

往事不再提,事還是要做滴。iOS10推出后,緊跟著Xcode8也推送了更新,細(xì)心者會發(fā)現(xiàn),Xcode8下iOS版本最低適配已變?yōu)閕OS8.0,加上Swift版本趨于穩(wěn)定,從某種意義上講,Swift的時(shí)代正式開啟,替代Objective-C怕也只是時(shí)間問題。當(dāng)然,在這之前,我們也應(yīng)做好準(zhǔn)備。今年越來的越多的公司,也開始了Swift和Objective-C混編。

我們今天就來看看兩者混編中的一些注意事項(xiàng)及問題:

混編

混編也無非兩種情況,

  • 在Objective - C工程或者文件使用Swift的文件;
  • 在Swift工程或者文件使用Objective - C文件

在混編的過程中最重要的兩個(gè)文件:
1. 橋接文件:
橋接文件“ProjectName-Bridging-Header.h”,在首次創(chuàng)建其他文件的時(shí)候,會自動生成。如果不小心刪除后,也可以手動添加,不過名字必須是“ProjectName-Bridging-Header.h”頭文件(名稱組成:工程名-Bridging-Header.h)。如果名字記不清也可以自己新建Header file后,在Targets-->Build Settings-->Swift Compiler - General-->Objective-C Bridging Header配置文件路徑,這個(gè)文件主要是Swift使用OC類時(shí)使用。

2. Objective-C Generated Interface Header Name文件
這個(gè)文件是混編時(shí),系統(tǒng)生成的Swift文件對應(yīng)的Objective-C的頭文件,具體可以在Targets-->Build Settings-->Swift Compiler - General-->Objective-C Generated Interface Header Name進(jìn)行配置,默認(rèn)文件名是工程名-Swift.h,一般不做改動。

在Objective - C工程或者文件使用Swift的文件

當(dāng)在OC文件中調(diào)用Swift文件中的類的時(shí)候,首先在OC文件中要加上 #import "
ProjectName-swift.h”(名字組成:工程名-swift)

這個(gè)文件雖然在工程中看不到,但是她真實(shí)存在,編譯后,你可以按住Command+單擊該文件名,就會看到具體生成的代碼。
引入后,具體類的使用,直接按照OC的方式使用即可。

在Swift工程或者文件使用Objective - C文件

當(dāng)在Swift中使用OC文件的時(shí)候,只需在橋接文件即projectName-Bridging-Header.h文件中引入需要的頭文件。
具體使用,按照對應(yīng)的Swift語法結(jié)構(gòu)來即可。

混編注意事項(xiàng)

對于需要混編的Swift類添加@objc聲明或繼承NSObject或NSObject的子類
class TestClass
{
// 屬性
// 實(shí)現(xiàn)
}

如果要在Objective-C類中使用TestClass類,應(yīng)當(dāng)使用@objc加以聲明,或者將TestClass繼承自NSObject或NSObject的子類,否則,引入ProductName-Swift.h之后,程序找不到對應(yīng)類。

使用第三方Framework
  • 設(shè)置: target-->build setting -->Packaging -->Defines Module為 “Yes”;
  • 然后,配置文件Target -> Build Phases -> Link Binary,添加要導(dǎo)入的Framework;
  • 最后,還是要配置橋接文件,比如要使用 abc-lib.framework庫中的 abc.h 就要這樣配置:#import"abc-lib/abc.h";
Subclass子類問題

對于自定義的類而言,Objective-C的類,不能繼承自Swift的類,即要混編的OC類不能是Swift類的子類。反過來,需要混編的Swift類可以繼承自O(shè)C的類。 注解

OC宏文件

如Swift文件要使用OC中定義的宏,只能使用常量簡單宏文件。

Swift獨(dú)有特性

Swift中有許多OC沒有的特性,比如,Swift有元組、為一等公民的函數(shù)、還有特有的枚舉類型。所以,要使用的混編文件要注意Swift獨(dú)有屬性問題。

案例之Swift中使用OC的block

*Swift 2. **:Swift中使用Closure不能使用Block作為屬性進(jìn)行傳值,必須是初始化方法或函數(shù)。
Objective-C文件中

#import <UIKit/UIKit.h>

typedef void (^Myblock)(NSString *arg); 

@interface FirViewController : UIViewController 
//@property (copy, nonatomic) Myblock myBlock; 
// Swift 2.*版本,這種作為公共參數(shù)的形式,如果在Swift類中去回調(diào)的話,是有問題的。提示沒有初始化方法,所以使用下面的以Block為參數(shù)的方法 

- (void)transValue:(Myblock) block;

@end 

下面是.m文件

#import "FirViewController.h" 
@implementation FirViewController 

- (void)viewDidLoad 
{ 
    [super viewDidLoad]; 
    self.view.backgroundColor = [UIColor whiteColor]; 
} 

- (void)transValue:(Myblock)block
{ 
    if (block) 
    { 
        block(@"firBack"); 
    } 
} 
@end 

在Swift文件回調(diào):

在Swift使用OC的類時(shí),首先在橋接文件中聲明oc的頭文件
工程名-Bridging-Header.h這是創(chuàng)建Swift工程的情況下

import UIKit 
class ViewController: UIViewController 
{ 
   override func viewDidLoad() 
    { 
        super.viewDidLoad() 
        self.view.backgroundColor = UIColor.whiteColor() 
    } 
    @IBOutlet weak var goFirst: UIButton! 
    @IBAction func goFirstAction(sender: AnyObject) 
    { 
        let firVC:FirViewController = FirViewController() 
        firVC. transValue { ( arg:String !) -> Void in 
            self.aBtn?.setTitle(arg, forState: UIControlState.Normal)
        } 
        self.navigationController?.pushViewController(firVC, animated: true) 
    } 

本文已在版權(quán)印備案,如需轉(zhuǎn)載請?jiān)诎鏅?quán)印獲取授權(quán)。
獲取版權(quán)

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容