欢迎访问移动开发之家(rcyd.net),关注移动开发教程。移动开发之家  移动开发问答|  每日更新
页面位置 : > > > 内容正文

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

来源: 开发者 投稿于  被查看 40889 次 评论:231

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


目录
  • 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相互调用的资料请关注3672js教程其它相关文章!

      您可能感兴趣的文章:
      • Flutter iOS开发OC混编Swift动态库和静态库问题填坑
      • swift framework使用OC 代码两种方式示例
      • swift闭包和OC block类型的使用
      • IOS开发OC代码中创建Swift编写的视图控制器
      • Swift中的协议(protocol)学习教程
      • Swift、Objective-C、Cocoa混合编程设置指南

      用户评论