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相互调用的资料请关注我们其它相关文章!

时间: 2022-08-29

Swift、Objective-C、Cocoa混合编程设置指南

Swift 被设计用来无缝兼容 Cocoa 和 Objective-C .在 Swift 中,你可以使用 Objective-C 的 API(包括系统框架和你自定义的代码),你也可以在 Objective-C中 使用 Swift 的 API.这种兼容性使 Swift 变成了一个简单.方便并且强大的工具集成到你的 Cocoa 应用开发工作流程中. 这篇指南包括了三个有关兼容性的重要方面方便你更好地利用来开发 Cocoa 应用: 互用性 使你将 Swift 和 Objective-C 相接合,允许在

Swift中的协议(protocol)学习教程

一.引言 协议约定了一些属性与方法,其作用类似Java中的抽象类,Swift中类型通过遵守协议来实现一些约定的属性和方法.Swift中的协议使用protocol关键字来声明.Swift中的协议还有一个十分有意思的特性,协议可以通过扩展来实现一些方法和附加功能. 二.在协议中定义属性和方法 协议中定义的属性只约定名称和类型,在具体类型的实现中,其可以是存储属性也可以是计算属性,协议中还需要指定属性是可读的还是可读可写的.示例代码如下: protocol MyPortocol { //定义实例属性

Flutter&nbsp;iOS开发OC混编Swift动态库和静态库问题填坑

目录 引言 OC接入Swift 插件 静态库和 Framework 区别 新的问题: non-modular heade 不能在Framework Module中使用非Modular 的 Header 引言 Flutter 在 iOS 上的编译问题相信大家多多少少遇到过,不知道大家在搜索这方便的问题时,得到的答案是不是让你 clean 或者 install 多几次,很多时候就算解决完问题,也是处于薛定谔的状态,所以本篇也简单记录下 Flutter 开发中,OC 混编 Swift 遭遇动态库和静态

IOS开发OC代码中创建Swift编写的视图控制器

IOS开发OC代码中创建Swift编写的视图控制器 背景 近日在和一群朋友做项目,我和另一位同学负责iOS客户端,我是一直使用OC的,而他只会Swift,因此在我们分工协作之后,就需要把代码合在一起,这就牵扯到如何在TabbarController中添加一个swift创建的子控制器的问题. 解决 首先在一个OC项目中新建一个Swift类,继承自UITableViewController,并且修改其view背景色,方便后续测试. import UIKit class ESSwiftTableVie

swift闭包和OC block类型的使用

之前看过一段swift,一直不知道OC中的block,即swift中的闭包是怎么实现的.今天就在网上搜索了一下,同时对比了一下OC中block类型的实现方法,然后写了一个Demo测试一下. 使用说明: swift版本 1.声明类型 typealias hideShowView = (Int) -> Void 2.声明属性 var muFunc:hideShowView? 3.传递参数 func didSelectedToHideView(hideFunc:@escaping (Int)->Vo

swift framework使用OC 代码两种方式示例

目录 引言 第一种: 第二种: 1. 新建一个module.modulemap文件: 2.设置module.modulemap引用路径 3.import module 引言 swift 在工程中可以通过桥接文件使用OC代码,但是在framework中是不支持使用桥接文件的,下面是是swift framework 使用 OC 代码的两种方式. 第一种: 在target->Build Phases->Headers中,将引入的oc文件由project移动到public中,再在公共.h文件中引入即可

android绘制圆形图片的两种方式示例

android绘制圆形图片的两种方式 看下效果先 下面有完整的示例代码 使用BitmapShader(着色器) 我们在绘制view 的时候 就是小学上美术课 用水彩笔在本子上画画 使用着色器绘制圆形图片最简单的理解方式 就是把bitmap当做一种颜色 设置给paint ,paint都已经有颜色了 你想让它方了,圆了,扁了 还不是看你心情 canvas调用那个方法咯 实现的大致思路如下: 1. 创建一个类 继承imageView 重写onDraw() 2. 获取到bitmap图片 3. 计算图片的

Python 多线程,threading模块,创建子线程的两种方式示例

本文实例讲述了Python 多线程,threading模块,创建子线程的两种方式.分享给大家供大家参考,具体如下: GIL(全局解释器锁)是C语言版本的Python解释器中专有的,GIL的存在让多线程的效率变低(哪个线程抢到锁,就执行哪个线程).在IO密集型程序中,多线程依然比单线程效率高(GIL通过IO阻塞自动切换多线程). 解决GIL(全局解释器锁)的问题的三种方法: 1.不要用C语言版本的Python解释器. 2.让子线程运行其他语言代码(例如:主线程运行Python代码,子线程运行C语言

PHP实现无限极分类的两种方式示例【递归和引用方式】

本文实例讲述了PHP实现无限极分类的两种方式.分享给大家供大家参考,具体如下: 面试的时候被问到无限极分类的设计和实现,比较常见的做法是在建表的时候,增加一个PID字段用来区别自己所属的分类 $array = array( array('id' => 1, 'pid' => 0, 'name' => '河北省'), array('id' => 2, 'pid' => 0, 'name' => '北京市'), array('id' => 3, 'pid' =>

t-sql清空表数据的两种方式示例(truncate and delete)

复制代码 代码如下: TRUNCATE TABLE (Transact-SQL) Removes all rows from a table without logging the individual row deletions. TRUNCATE TABLE is similar to the DELETE statement with no WHERE clause; however, TRUNCATE TABLE is faster and uses fewer system and t

Hibernate中获取Session的两种方式代码示例

Session:是应用程序与数据库之间的一个会话,是Hibernate运作的中心,持久层操作的基础.对象的生命周期/事务的管理/数据库的存取都与Session息息相关. Session对象是通过SessionFactory构建的,下面举个例子来介绍Hibernate两种获取session的方式. 日志,是编程中很常见的一个关注点.用户在对数据库进行操作的过程需要将这一系列操作记录,以便跟踪数据库的动态.那么一个用户在向数据库插入一条记录的时候,就要向日志文件中记录一条记录,用户的一系列操作都要在

.NET实现定时发送邮件代码(两种方式)

有时候我们或许会遇到想在某一个时刻给别人发送一封邮件,就像是在生日的时候,但是我们又怕到时候忘记了,这时就应该 使用发送定时邮件的功能,但是这个定时发送邮件功能是怎么实现的呢?下面用两种方式实现.net定时发送邮件代码,具体请看下面内容. 实现思路.需求添加一个全局应用程序类Global.asax 代码会在访问网站时运行 Global.asax代码: void Application_Start(object sender, EventArgs e) { // 在应用程序启动时运行的代码 Sys

Flutter以两种方式实现App主题切换的代码

概述 App主题切换已经成为了一种流行的用户体验,丰富了应用整体UI视觉效果.例如,白天夜间模式切换.实现该功能的思想其实不难,就是将涉及主题的资源文件进行全局替换更新.说到这里,我想你肯定能联想到一种设计模式:观察者模式.多种观察对象(主题资源)来观察当前主题更新的行为(被观察对象),进行主题的更新.今天和大家分享在 Flutter 平台上如何实现主题更换. 效果 实现流程 在 Flutter 项目中,MaterialApp组件为开发者提供了设置主题的api: const MaterialAp

Springboot整合MongoDB进行CRUD操作的两种方式(实例代码详解)

1 简介 Springboot是最简单的使用Spring的方式,而MongoDB是最流行的NoSQL数据库.两者在分布式.微服务架构中使用率极高,本文将用实例介绍如何在Springboot中整合MongoDB的两种方法:MongoRepository和MongoTemplate. 代码结构如下: 2 项目准备 2.1 启动MongoDB实例 为了方便,使用Docker来启动MongoDB,详细指导文档请参考:基于Docker的MongoDB实现授权访问的方法,这里不再赘述. 2.2 引入相关依赖

Spring boot调用Oracle存储过程的两种方式及完整代码

前言 因工作需要将公司SSH项目改为Spingboot项目,将项目中部分需要调用存储过程的部分用entityManagerFactory.unwrap(SessionFactory.class).openSession()来获取Session实现后发现项目访问数据库超过十次就会挂掉,原因是Springboot连接池数量默认为10,猜测是每次访问数据库后连接未释放导致的,手动关闭session后问题解决. 解决问题的过程中又发现了另外两种调用方式: 直接用EntityManager的createS

MyBatis批量插入数据到Oracle数据库中的两种方式(实例代码)

一.mybatis批量插入数据到Oracle中的两种方式: 第一种: <insert id="addList" parameterType="java.util.List" useGeneratedKeys="false"> INSERT ALL <foreach item="item" index="index" collection="list"> INTO