前言 Swift已推出數年,與Objective-C相比Swift的語言機制及使用簡易程度上更接地氣,大大降低了iOS入門門檻。當然這對新入行的童鞋沒來講,的確算是福音,但對於整個iOS編程從業者來講,真真是,曾幾何時“高大上”,轉瞬之間“矮矬窮”。再加上培訓班橫行,批量批發之下,iOS再也看不到當 ...
前言
Swift已推出數年,與Objective-C相比Swift的語言機制及使用簡易程度上更接地氣,大大降低了iOS入門門檻。當然這對新入行的童鞋沒來講,的確算是福音,但對於整個iOS編程從業者來講,真真是,曾幾何時“高大上”,轉瞬之間“矮矬窮”。再加上培訓班橫行,批量批發之下,iOS再也看不到當年的輝煌。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) }
Swift和Objective-C語言的混編就分享這麼多,希望大神給出寶貴的意見!(@。ε。@)