混编方案
Swift调OC比较简单,因为Swift覆盖了OC的大部分功能。没有覆盖的部分,官方也会给出解决办法。而Swift却有一些特有的功能是OC没有的,所以当OC在调用Swift的时候,这些Swift特有的功能被引入后,就会导致出错。因为OC并不支持。
OC调用Swift,需要先#import "工程名-Swift.h"
- 当Swift的类不是继承自NSObject时,OC就无法调用。因为OC的类,最终都继承自NSObject。给类加上
: NSObject
。系统会给继承自NSObject的类,自动加上@objc
修饰,除了private修饰的方法和属性。 - 对于不是继承自NSObject的类,需要在方法或属性前加上
@objc
。在Swift4之后,即使是继承自NSObject的类,也需要显式添加@objc
才能访问。在此之前是默认添加的。 - Swift中的枚举。OC不能使用,OC中的枚举实际都是
int
型
需要要继承Int
,加上@objc
。
enum MethodType {
case POST
case GET
case DELETE
}
需改成
@objc enum MethodType: Int {
case POST
case GET
case DELETE
}
- Swift中是结构体类型。而OC中是基本数据类型。所以如果Swift是表示基本数据类型的时候,OC可以转换。如果表示的是结构体才有的功能。那么oc无法转换。
例:
@objc var a: String?
这里的变量a可以转换,查看一下工程名-Swift.h文件,的确生成了OC头文件:
@property (nonatomic, copy) NSString * _Nullable a;
如果将这里的a类型改为Int会怎么样:
@objc var a: Int?
会报错:Property cannot be marked @objc because its type cannot be represented in Objective-C
因为当Int
成为可选类型的时候,这里的Int
已经不光是表示一个简单的Int
类型了,而是带有明显的结构体属性,因为是可选,即表示,可能是nil
,而OC里的Int类型或者NSInteger都是基本数据类型,不能为nil
。
所以,要解决这个问题,只能在Swift中避免将OC中的基本数据类型设置为可选类型。
- OC不支持Swift中的泛型,虽然从Xcode7,OC就引入了泛型的支持,但是这个泛型,我们称作轻量级泛型,和Swift中的重量级泛型还是对不上,当然,硬是要去对接,苹果应该也能做,但毕竟不是同一个东西,强行对接反而不好。
参考文献:
OC调Swift的问题记录
常见问题
-
oc调用Swift,import *-Swift.h后,无法调用Swift的类方法
Swift的对应类名前增加 @objcMembers(在class 类名上方)
-
Swift调用oc,oc的pch文件不可用
在项目target 的 build settings里,找到prefix header。添加pch的位置,例如$(SRCROOT)/Common/tool/Prefix.pch
-
OC语法
-
switch case语句
因为switch case 中是不能定义对象的,所以需要在大括号内定义的对象。否则,会出现Cannot jump from switch statement to this case label
的错误
{
case 0:
{
// 开始录音
self.recordStatus = 1;
[self.rippleView startAnimationWithCircleNumber:2];
[ALAudioPlayerTool.share startRecord];
break;
}
case 1:
{
// 待播放
self.recordStatus = 2;
[self.rippleView stopAnimation];
[self.audioRecordButton setImage:[UIImage imageNamed:@"icon_learn_audio_play"] forState:UIControlStateNormal];
[ALAudioPlayerTool.share stopRecord];
self.resetRecordButton.hidden = NO;
break;
}
default:
break;
}