Swift开发是一种大趋势了,每年Apple都会发布新的版本。但是它的第三方库还是没有Objective-C的多,而且想用老项目中的工具类或框架,该怎么办呢?
这就需要Swift&Objective-C混编互调了,但是在Objective-C的编译器主要可以识别以下几种扩展名的文件:
- .m文件,可以编写Objective-C代码或者C语言代码;
- .cpp文件,C++文件,只能识别C++或者C语言代码;
- .mm,主要用于编写C++和Objective-C混编的代码,可以同时识别Objective-C、C和C++代码。
那么该怎么让Swift&Objective-C混编互调呢?在Apple发布Swift的时候已经提供了方案,在iOS8以后Apple给出了这两种语言之间的桥接方案,简单来说就是在Swift工程中,通过提示创建的Bridging头文件可以将Objective-C文件和Swift文件衔接在一起,从而可以在Objective-C文件中引用Swift类,或者在Swift文件中引用Objective-C的类。
下面我们一起来具体操作一下:
1. 创建一个Swift工程,选择Single View App
2. 先新建一个swift类
新建类SwiftObjct.swift,包含一个属性,一个实例方法,一个类方法:
import UIKit
class SwiftObjct: NSObject
{
var name = "swiftObject"
func instanceMethod( ) -> Void
{
print("swift instance method")
}
class func classMethod( ) -> Void
{
print("swift class method")
}
}
3. 创建第一个OC类时会提示创建一个bridging header文件
会自动生成一个头文件 SwiftObjectiveC-Bridging-Header.h(前缀是工程名)
再创建一个Objective-C类OCObject,同样包含一个属性,一个实例方法,一个类方法:
#import <Foundation/Foundation.h>
NS_ASSUME_NONNULL_BEGIN
@interface OCObject : NSObject
@property (nonatomic,strong) NSString *name;
-(void)instanceMethod;
+(void)classMethod;
@end
NS_ASSUME_NONNULL_END
4. Swift类调用Objective-C的类
在工程自动生成的类ViewController.swift中调用我们刚创建的Objective-C的类OCObject
import UIKit
class ViewController: UIViewController {
override func viewDidLoad() {
super.viewDidLoad()
let ocObject = OCObject();
ocObject.instanceMethod()
OCObject.classMethod()
}
}
此时编译器会报错Use of unresolved identifier 'OCObject'
,原因是我们还没有把Objective-C的类和Swift关联起来。在创建第一个Objective-C类OCObject的时候生成的头文件SwiftObjectiveC-Bridging-Header.h,就是连接它们的桥梁。
在头文件中导入刚创建的Objective-C类OCObject:
#import "OCObject.h"
然后一定要command+B编译通过一下,此时ViewController.swift调用的OCObject就不再报错了。
5. Objective-C类调用Swift类
上一步建立的连接之后,接下来我们在OCObject.m实现它的两个方法,并调用SwiftObjct:
-(void)instanceMethod
{
SwiftObjct *swiftObject = [[SwiftObjct alloc] init];
[swiftObject instanceMethod];
NSLog(@"oc instance method, swiftObjct.name=%@", swiftObject.name);
}
+(void)classMethod
{
[SwiftObjct classMethod];
NSLog(@"oc class method");
}
此时还会报错Use of undeclared identifier 'SwiftObjct'
,这个地方就还需要导入一个新的文件:
// 名字前缀是工程名
#import "SwiftObjectiveC-Swift.h"
你会发现,在工程中搜不到此头文件,这是因为这个类时隐藏的,工程自动生成的,它的作用就是对工程中所有swift类文件进行了向Objective-C语言的翻译。所以导入这个头文件之后,上面的代码就不会报刚才的错了。但是又报新的错误了,SwiftObjct类是识别了,但是不识别它的方法和属性:
这是因为Swift4.0以后,暴露给Objective-C类调用的swift方法和属性都要在前面加上修饰词:@objc,否则Objective-C类无法找到对应的Swift方法和属性。另外需要注意定义后一定要command+B编译通过一下工程,才能正常在Objective-C文件中调用swift属性和方法。
所以对SwiftObjct.swift修改如下:
import UIKit
class SwiftObjct: NSObject
{
@objc var name = "swiftObject"
@objc func instanceMethod( ) -> Void
{
print("swift instance method")
}
@objc class func classMethod( ) -> Void
{
print("swift class method")
}
}
然后command+B编译过后,报错都没有了,并正常打印:
swift instance method
2019-08-12 18:06:09.617260+0800 SwiftObjectiveC[19281:400024] oc instance method, swiftObjct.name=swiftObject
swift class method
2019-08-12 18:06:09.617474+0800 SwiftObjectiveC[19281:400024] oc class method
注意:
1、Swift类中不需要import头文件,因为它们的作用域是全局的;
2、SwiftObjectiveC-Swift.h文件是隐藏的,它对工程中所有swift类文件进行了向Objective-C语言的翻译;
3、修改Swift类后一定要command+B编译通过一下。