IOS开发Swift 与 OC相互调用详解

Nita ·
更新时间:2024-09-20
· 1819 次阅读

目录

1、创建桥接文件

2、Swift调用OC

NS_SWIFT_NAME、NS_SWIFT_UNAVAILABLE

NS_REFINED_FOR_SWIFT

规则

3、OC调用Swift

4、坑点

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相互调用的资料请关注软件开发网其它相关文章!



Swift oc 调用 IOS

需要 登录 后方可回复, 如果你还没有账号请 注册新账号