首頁 > 軟體

IOS開發Swift 與 OC相互呼叫詳解

2022-08-29 22:05:05

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 上不會有對應的類出現

以上就是IOS開發Swift 與 OC相互呼叫詳解的詳細內容,更多關於Swift OC相互呼叫的資料請關注it145.com其它相關文章!


IT145.com E-mail:sddin#qq.com