IOS怎么开发Swift与OC相互调用

寻技术 Swift 2023年09月08日 93

本篇内容主要讲解“IOS怎么开发Swift与OC相互调用”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“IOS怎么开发Swift与OC相互调用”吧!

    1、创建桥接文件

    在创建另一种语言的文件时XCode会提示创建项目名-Bridging-Header.h的桥接文件

    2、Swift调用OC

    1.创建OC文件

    #import "MyViewController.h"
    @interface MyViewController ()
    @end
    @implementation MyViewController
    - (void)viewDidLoad {
        [super viewDidLoad];
        self.title = @"OC";
        self.view.backgroundColor = UIColor.cyanColor;
        UILabel *lbl = [UILabel new];
        lbl.text = @"OC的标签";
        lbl.backgroundColor = UIColor.redColor;
        lbl.frame = CGRectMake(100, 100, 150, 50);
        [self.view addSubview:lbl];
    }

    2.桥接文件:项目名-Bridging-Header.h 文件中要将想要使用的 OC的.h文件导入

    #import "MyViewController.h"

    3.在Swift文件中调用

    import UIKit
    class ViewController: UIViewController {
        override func viewDidLoad() {
            super.viewDidLoad()
            // Do any additional setup after loading the view.
            title = "Swift"
            view.backgroundColor = .white
            let btn = UIButton(type: .custom)
            btn.frame = CGRect(x: 150, y: 150, width: 150, height: 100)
            btn.setTitle("点击跳转", for: .normal)
            btn.backgroundColor = .green
            view.addSubview(btn)
            btn.addTarget(self, action: #selector(didClickBtn), for: UIControl.Event.touchUpInside)
        }
        @objc func didClickBtn() {
            let myVC = MyViewController()
            navigationController?.pushViewController(myVC, animated: true)
        }
    }

    Button执行的方法要用 @objc 修饰

    NS_SWIFT_NAME、NS_SWIFT_UNAVAILABLE

    • NS_SWIFT_NAME(替换名):重命名在Swift中的名称,可用来进行方法名隐藏

    • NS_SWIFT_UNAVAILABLE(_msg):Swift中不可见,不能使用

    // OC的MyViewController.h文件
    #import <UIKit/UIKit.h>
    NS_ASSUME_NONNULL_BEGIN
    @interface MyViewController : UIViewController
    // 将method1方法在Swift中替换成swiftMethod()方法
    - (void)method1 NS_SWIFT_NAME(swiftMethd());
    // 将method2方法再Swift中隐藏
    - (void)method2 NS_SWIFT_UNAVAILABLE("Swift中该方法不可调用");
    @end
    NS_ASSUME_NONNULL_END
    class ViewController: UIViewController {
        override func viewDidLoad() {
            ......
        }
        @objc func didClickBtn() {
            let myVC = MyViewController()
            // 在Swift中找不到OC的method1与method2方法,只有一个改了名的swiftMethod方法
            myVC.swiftMethod()
            navigationController?.pushViewController(myVC, animated: true)
        }
    }

    NS_REFINED_FOR_SWIFT

    在Swift中调用OC的接口有时发现并不符合Swift的语法规范或者使用起来会比较别扭,这个时候可以使用NS_REFINED_FOR_SWIFT宏定义 来对OC的接口进行升级改造

    规则

    NS_REFINED_FOR_SWIFT 可用于方法和属性,添加了 NS_REFINED_FOR_SWIFT 的 Objective-C API 在导入到 Swift 时,具体的 API 重命名规则如下:

    对于 初始化方法,在其第一个参数标签前面加 "__"

    // Objective-C API
    - (instancetype)initWithClassName:(NSString *)name NS_REFINED_FOR_SWIFT;
    // In Swift
    init(__className: String)

    对于 其它方法,在其基名前面加 "__"

    // Objective-C API
    - (NSString *)displayNameForMode:(DisplayMode)mode NS_REFINED_FOR_SWIFT;
    // In Swift
    func __displayNameForMode(mode: DisplayMode) -&gt; String

    下标方法将被视为任何其它方法,在方法名前面加 "__"(而不是作为 Swift 下标导入)

    其他声明将在其名称前加上 "__",例如属性

    // Objective-C API
    @property DisplayMode mode NS_REFINED_FOR_SWIFT;
    // In Swift
    var __mode: DisplayMode { get set }

    注意:NS_REFINED_FOR_SWIFT 和 NS_SWIFT_NAME 一起用的话,NS_REFINED_FOR_SWIFT 不生效,而是以 NS_SWIFT_NAME 指定的名称重命名 Objective-C API

    3、OC调用Swift

    创建Swift文件

    import Foundation
    // 必须继承于 NSObject
    class Person: NSObject {
        // 想公开给OC的要使用 @objc 修饰
        @objc var name: String
        @objc var age : Int
        @objc init(name: String, age: Int) {
            self.name = name
            self.age = age
        }
    }
    • 必须继承于 NSObject,类、结构体等才会公开给OC

    • 必须使用 @objc 修饰,属性、方法等才会公开给OC

    在Swift文件中引入项目名-Swift.h文件,然后使用Swift内容

    #import "MyViewController.h"
    #import "SwiftAndOC-Swift.h"
    @interface MyViewController ()
    @end
    @implementation MyViewController
    - (void)viewDidLoad {
        [super viewDidLoad];
        Person *p = [[Person alloc] initWithName:@"LZ" age:18];
        NSLog(@"%@",p.name);
    }

    4、坑点

    • OC类不能继承于Swift类,但Swift类可以继承于OC类

    • Swift中没有宏定义:

    常量宏用let参数代替

    无参变量宏可以用"只读属性"代替也可用函数代替

    变量宏用函数代替

    • 要给OC用的内容不要用Swift独有特性书写(比如元组)

    • 如果OC通过pod的形式集成Swift,需要在 Swift的类上也要声明Public,否则在对应的 项目名-Swift.h 上不会有对应的类出现

    关闭

    用微信“扫一扫”