Swift&Objective-C混编互调

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

Swift&Objective-C混编互调_第1张图片

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文件

Swift&Objective-C混编互调_第2张图片

会自动生成一个头文件 SwiftObjectiveC-Bridging-Header.h(前缀是工程名)

再创建一个Objective-C类OCObject,同样包含一个属性,一个实例方法,一个类方法:

#import 

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类是识别了,但是不识别它的方法和属性:

Swift&Objective-C混编互调_第3张图片

这是因为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编译通过一下。

你可能感兴趣的:(iOS编程,笔记)