天天看點

Swift和Objective-C混編的注意啦

前言

Swift已推出數年,與Objective-C相比Swift的語言機制及使用簡易程度上更接地氣,大大降低了iOS入門門檻。當然這對新入行的童鞋沒來講,的确算是福音,但對于整個iOS程式設計從業者來講,真真是,曾幾何時“高大上”,轉瞬之間“矮矬窮”。再加上教育訓練班橫行,批量批發之下,iOS再也看不到當年的輝煌。

Swift和Objective-C混編的注意啦

往事不再提,事還是要做滴。iOS10推出後,緊跟着Xcode8也推送了更新,細心者會發現,Xcode8下iOS版本最低适配已變為iOS8.0,加上Swift版本趨于穩定,從某種意義上講,Swift的時代正式開啟,替代Objective-C怕也隻是時間問題。當然,在這之前,我們也應做好準備。今年越來的越多的公司,也開始了Swift和Objective-C混編。

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

混編

混編也無非兩種情況,

在Objective - C工程或者檔案使用Swift的檔案;
在Swift工程或者檔案使用Objective - C檔案      

在混編的過程中最重要的兩個檔案:

1.橋接檔案:

橋接檔案“ProjectName-Bridging-Header.h”,在首次建立其他檔案的時候,會自動生成。如果不小心删除後,也可以手動添加,不過名字必須是“ProjectName-Bridging-Header.h”頭檔案(名稱組成:工程名-Bridging-Header.h),如果名字記不清也可以自己建立Header file後,在Targets→Build Settings→Swift Compiler - General→Objective-C Bridging Header配置檔案路徑,這個檔案主要是Swift使用OC類時使用。

2.Objective-C Generated Interface Header Name檔案

這個檔案是混編時,系統生成的Swift檔案對應的Objective-C的頭檔案,具體可以在Targets→Build Settings→Swift Compiler - General→Objective-C Generated Interface Header Name進行配置,預設檔案名是工程名-Swift.h,一般不做改動。

在Objective - C工程或者檔案使用Swift的檔案

當在OC檔案中調用Swift檔案中的類的時候,首先在OC檔案中要加上 #import “

ProjectName-swift.h”(名字組成:工程名-swift)

這個檔案雖然在工程中看不到,但是她真實存在,編譯後,你可以按住Command+單擊該檔案名,就會看到具體生成的代碼。

引入後,具體類的使用,直接按照OC的方式使用即可。

在Swift工程或者檔案使用Objective - C檔案

當在Swift中使用OC檔案的時候,隻需在橋接檔案即projectName-Bridging-Header.h檔案中引入需要的頭檔案。

具體使用,按照對應的Swift文法結構來即可。

混編注意事項

對于需要混編的Swift類添加@objc聲明或繼承NSObject或NSObject的子類

class TestClass
{
// 屬性
// 實作
}      

如果要在Objective-C類中使用TestClass類,應當使用@objc加以聲明,或者将TestClass繼承自NSObject或NSObject的子類,否則,引入ProductName-Swift.h之後,程式找不到對應類。

使用第三方Framework

設定: target-->build setting -->Packaging -->Defines Module為 “Yes”;
然後,配置檔案Target -> Build Phases -> Link Binary,添加要導入的Framework;
最後,還是要配置橋接檔案,比如要使用 abc-lib.framework庫中的 abc.h 就要這樣配置:#import"abc-lib/abc.h";      

Subclass子類問題

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

OC宏檔案

如Swift檔案要使用OC中定義的宏,隻能使用常量簡單宏檔案。

Swift獨有特性

Swift中有許多OC沒有的特性,比如,Swift有元組、為一等公民的函數、還有特有的枚舉類型。是以,要使用的混編檔案要注意Swift獨有屬性問題。

案例之Swift中使用OC的block

Swift中使用Closure不能使用Block作為屬性進行傳值,必須是初始化方法或函數。

Objective-C檔案中:

#import <UIKit/UIKit.h>

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

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

- (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檔案回調:

在Swift使用OC的類時,首先在橋接檔案中聲明oc的頭檔案

工程名-Bridging-Header.h這是建立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) 
    }