如何实现 Objective-C 与 Swift 混编
在混编过程中分为两种情况
- Objective-C 导入 swift
- swift 导入 Objective-C
下面进行分开讲解
将 Objective-C 导入 Swift
在swift 中访问你到 Objective-C 类以及其他声明
概述
你可以在一个项目中同时使用Objective-C和Swift文件,不管项目最初使用的是哪种语言。这使得创建混合语言的应用程序和 framework target 就像创建用单一语言编写的应用程序或 framework target 一样简单。
在混合语言 target 中使用Swift代码中的Objective-C声明的过程略有不同,这取决于你是在编写 app 还是 framework 。下面将描述这两个过程。
导入app target 中的代码
要在同一个app目标中导入一组Objective-C文件到Swift代码中,你需要依靠一个Objective-C桥接头文件将这些文件暴露给Swift。当你将一个Swift文件添加到现有的Objective-C应用程序或将一个Objective-C文件添加到现有的Swift应用程序时,Xcode提供创建这个头文件。
如果您接受,Xcode将创建桥接头文件以及您正在创建的文件,并使用您的“项目名称”后跟“- bridge - header .h”来命名它。或者,您也可以通过以下选择创建桥接头
File > New > File > [operating system] > Source > Header File.
编辑桥接头,将你的Objective-C代码暴露给你的Swift代码:
- 在Objective-C桥接头中,导入你想要公开给Swift的每个Objective-C头。
- 在 Build Settings 中,在 Swift Compiler - General 下,确保Objective-C桥接头构建设置有一个到桥接头文件的路径。路径应该相对于你的项目,类似于你的信息。plist路径在生成设置中指定。在大多数情况下,您不需要修改此设置。
任何列在桥接头中的公共Objective-C头文件对Swift都是可见的。Objective-C声明可以自动从目标内的任何Swift文件中获得,不需要import语句。用与系统类相同的Swift语法使用自定义Objective-C代码中的类和其他声明。
在 framework target 内导入代码
要在与Swift代码相同的框架目标文件中使用Objective-C声明,你需要将这些文件导入到 Objective-C umbrella header 中——framework 的主头文件。通过umbrella header 导入你的Objective-C文件:
- 在Build Settings下打包,确保framework target 的 Defines Module setting 设置为Yes。
- 在 umbrella header 中,导入你想要公开给Swift的每个Objective-C头。
Swift 可以看到你在 umbrella header 中公开的每一个头文件。该 framework 中的 Objective-C 文件的内容可以自动从该 framework target 内的任何 Swift 文件中获得,不需要 import 语句。用与系统类相同的 Swift 语法使用 Objective-C 代码中的类和其他声明。
将 Swift 导入 Objective-C
从你的Objective-C代码库中访问Swift类型和声明。
概述
通过导入xcode生成的头文件,您可以在项目的Objective-C代码中使用Swift中声明的类型。这个文件是一个Objective-C头文件,它在你的 target 中声明了Swift接口,你可以把它想象成Swift代码的 umbrella header 文件。您不需要做任何特殊的事情来创建生成的头文件—只需导入它以在您的Objective-C代码中使用它的内容。
头文件的名称是由您的项目名称生成的,后面跟着“-Swift.h”。默认情况下,此名称与您的产品名称相同,使用下划线(_)替换任何非字母数字字符。如果名称以数字开头,则第一个数字将被下划线替换。
将Swift声明导入到Objective-C代码的过程略有不同,这取决于你是在编写应用程序还是框架。下面将描述这两个过程。
导入 app target 中的代码
当你创建一个app target时,你可以使用以下语法将Swift代码导入到同一个目标中的任何Objective-C .m文件中,并替换相应的名称:
#import "ProductModuleName-Swift.h"
默认情况下,生成的头包含Swift声明的接口,这些声明带有public或open修饰符。如果你的 app target 有一个 Objective-C 桥接头,那么生成的头还包括用内部修饰符标记的接口。用 private 或fileprivate修饰符标记的声明不会出现在生成的头文件中,也不会暴露给Objective-C运行时,除非它们被显式地标记为 @IBAction 、 @IBOutlet 或 @objc 属性。在单元测试目标内部,您可以访问导入的内部声明,就像它们是公共的一样,方法是在 product module import 语句之前加上@testing。
在生成的头文件中,Swift接口包含了对所有使用的Objective-C类型的引用,所以请确保首先为这些类型导入Objective-C头文件。
在构建app target 时,可以通过更改 product 模块名称,build setting为product模块提供自定义名称。Xcode在命名生成的头文件时使用这个名称。
导入 Framework Target 中的代码
要在与 Objective-C 代码相同的framework target 中导入一组Swift文件,请将xcode生成的Swift代码头文件导入到任何您想使用Swift代码的Objective-C .m文件中。
因为生成的头文件是 framework 公共接口的一部分,所以对于framework,只有标记了public或open修饰符的声明才会出现在生成的头文件中。用内部修饰符标记并在继承自Objective-C类的类中声明的方法和属性可以被Objective-C运行时访问。但是,它们在编译时不可访问,并且不会出现在为framework target 生成的头文件中。
在同一个框架内将Swift代码导入到Objective-C中:
- 在 Build Settings 下的 Packaging中, 确保将该framework target 的“Defines Module setting”设置为“YES”。
- 使用以下语法将该framework target 中的Swift代码导入到该目标中的任何Objective-C .m文件中,并替换相应的名称:
#import <ProductName/ProductModuleName-Swift.h>
在Objective-C头文件中使用前向声明包含Swift类
当Objective-C头文件中的声明引用来自同一目标的Swift类或协议时,导入生成的头会创建一个循环引用。为了避免这种情况,可以使用Swift类或协议的前向声明在Objective-C接口中引用它。
// MyObjcClass.h
@class MySwiftClass;
@protocol MySwiftProtocol;
@interface MyObjcClass : NSObject
- (MySwiftClass *)returnSwiftClassInstance;
- (id <MySwiftProtocol>)returnInstanceAdoptingSwiftProtocol;
// ...
@end
Swift类和协议的前向声明只能用作方法和属性声明的类型。