Swift和OC混编

2022-06-29  本文已影响0人  大刘

Created by 大刘 liuxing8807@126.com

OC和Swift的混编,分为两种情况:

  1. OC项目中嵌入Swift文件
  2. Swift项目中嵌入OC文件

先来看OC项目中嵌入Swift文件的情况

新建一基于OC的项目,比如:OC_project, 接下来我们创建一个Swift类: Person
在新建这个Swift类时,Xcode会提示生成桥接文件:

3.png

添加这个文件会创建一个OC和swift的混合target, 这样可以让OC和Swift两个语言都可以访问class, 这个文件的主要目的是暴露一些OC的header, 让Swift可以调用OC:

OC_project-Bridging-Header.h(让Swift可以调OC):

Use this file to import your target's public headers that you would like to expose to Swift.

// Person.swift
import Foundation

// 和java不同,这里public的类名Person和文件名可以不一样,但作为习惯,一般和文件名写成一样
// public可以省略,外面同样可以调得到
// 如果要OC调用到这个Swift类,@objc不可省略,而且这个类要直接或间接继承于NSObject
@objc public class Person : NSObject {
    // 为了让OC调用,方法同样需要@objc标示
    @objc
    func think() -> Void {
        print("person can think, it's important.")
    }
    
    @objc
    func eat() -> Void {
        print("person can eat, it's important.")
    }
    
    @objc
    func drink(_ water: String) -> Void {
        print("person can drink \(water), it's important")
    }
}

然后我们在OC的类里面调用这个Swift类:

#import "ViewController.h"
#import "OC_project-Swift.h" // 让OC可以调Swift, 格式:工程名-Swift.h

@interface ViewController ()

@end

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    
    // 依赖于Apple自动生成的"工程名-Swift.h"文件,本示例中是:OC_project-Swift.h
    // 其实在这个文件中声明了一些Swift到OC的转换后方法,因此我们可以以OC的调用方式调用Swift方法
    Person *p = [[Person alloc] init];
    NSLog(@"%@", p);
    [p think];
    [p eat];
    [p drink:@"咖啡"];
}

@end

注意,一般这样做就可以了,但Xcode有时候会抽疯的,因此如果有问题我们需要检查Xcode的设置:
1. Defines Module: YES

4.png

2. Product Module Name : Your Project Name

5.png

Make sure that your Product Module Name doesn't contain any special characters.

3. Install Objective-C Compatibility Header : YES

6.png

Once you've added *.swift file to the project this property will appear in Build Settings

4. Objective-C Generated Interface Header : YourProjectName-Swift.h

7.png

This header is auto generated by Xcode

5. Objective-C Bridging Header : $(SRCROOT)/yourProject-Bridging-Header.h

8.png

上面已经可以让OC调用Swift正常运作,下面我们依然基于这个OC项目,让Swift调用OC,在做这个之前我们需要知道一个概念:

在上面的示例中, 使用OC调Swift并不需要使用桥接文件, 但如果要让Swift调OC,则需要在这个文件中import要在Swift中使用的OC类的头文件。
接下来我们新建一个OC的Dog类,让Swift方法调用这个Dog类的方法。

在上面的示例中, 使用OC调Swift需要引入系统文件“工程名-Swift.h”,但如果要让Swift调OC,则需要在桥接文件OC_project-Bridging-Header.h中引入要在Swift中使用的OC类的头文件:

// -- OC_project-Bridging-Header.h --
#import "Dog.h"
-- Dog.h --
@interface Dog : NSObject

- (void)think;
- (void)eat:(NSString *)food;
@end

-- Dog.m --
@implementation Dog

- (void)think {
    NSLog(@"Dog can think, but it's different with person's think");
}

- (void)eat:(NSString *)food {
    NSLog(@"%@", [NSString stringWithFormat:@"dog can eat: %@, it's important", food]);
}

@end
// -- Person.swift --
import Foundation

@objc public class Person : NSObject {
    ...
   
    @objc
    func dogThink() -> Void {
        let dog: Dog = Dog()
        dog.think()
    }

    @objc
    func dogEat(_ food: String) -> Void {
        let dog: Dog = Dog()
        dog.eat(food)
    }
}
-- ViewController.m --
@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    
    // 依赖于Apple自动生成的‘工程名-Swift.h’文件
    Person *p = [[Person alloc] init];
    NSLog(@"%@", p);
    [p think];
    [p eat];
    [p drink:@"咖啡"];
    [p dogThink]; // 添加
    [p dogEat:@"狗粮"]; // 添加
}

@end

以上是基于OC项目嵌入Swift文件,接下来我们在Swift项目中嵌入OC文件

Swift项目中嵌入OC文件相对比较简单,新建一基于Swfift的项目,比如:Swift_project, 然后新建OC类,在新建的时候同样按照系统提示生成bridge文件。

如果Swift类要调用OC,只须在Swift_project-Bridging-Header.h#import相对应的OC类即可。如果OC类要调用Swift, 只需#import "Swift_project-Swift.h"即可。

示例:

-- Person.h --
@interface Person : NSObject

- (void)think;
- (void)dogThink;

@end

-- Person.m --
#import "Person.h"
#import "Swift_project-Swift.h"

@implementation Person

- (void)think {
    NSLog(@"think");
}

- (void)dogThink {
    Dog *d = [[Dog alloc] init];
    [d think];
}

@end
-- Dog.swift --
import UIKit

@objc
class Dog: NSObject {
    @objc
    public func think() -> Void {
        print("dong think, but different with person")
    }
}

-- Swift_project-Bridging-Header.h --
#import "Person.h"
-- ViewController.swift --
class ViewController: UIViewController {
    override func viewDidLoad() {
        super.viewDidLoad()
     
        let p: Person = Person()
        p.think()
        p.dogThink()
    }
}
上一篇 下一篇

猜你喜欢

热点阅读