Swift&Objective-C混编互调

Swift开发是一种大趋势了,每一年Apple都会发布新的版本。可是它的第三方库仍是没有Objective-C的多,并且想用老项目中的工具类或框架,该怎么办呢?swift

这就须要Swift&Objective-C混编互调了,可是在Objective-C的编译器主要能够识别如下几种扩展名的文件:bash

  • .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的类。框架

下面咱们一块儿来具体操做一下:ide

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,一样包含一个属性,一个实例方法,一个类方法:ui

#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的类OCObjectatom

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,就是链接它们的桥梁。spa

在头文件中导入刚建立的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
复制代码
注意:

一、Swift类中不须要import头文件,由于它们的做用域是全局的;
二、SwiftObjectiveC-Swift.h文件是隐藏的,它对工程中全部swift类文件进行了向Objective-C语言的翻译;
三、修改Swift类后必定要command+B编译经过一下。

欢迎加入iOS开发交流学习群,咱们一块儿共同窗习,共同成长

相关文章
相关标签/搜索