一、OC调用swift文件
这里创建一个名为Test的OC项目
在OC项目中创建一个swift文件Person.swift,会提示是否创建桥接文件。这个桥接文件是用于swift调用OC的,可以创建,若不创建则后续手动创建也是可以的。
编译器会为我们自动创建一个隐藏的Test-swift.h文件。
1、只要在OC文件里引入这个文件就可以使用swift文件。
2、这个文件名称默认是:Product Module Name-Swift.h
。
3、如果不希望使用默认名称,也可以自定义,在Build Settings中找到Object-C Generated interface Header Name,这里可以自定义用于引入的swift头文件名称,一般格式为:Name-Swift.h
,若修改了文件名称,需重新编译才可以识别到。注意,当工程名称中有横线时,会被自动转换为下划线。
4、xcode会根据swift代码生成对应的OC声明,写入到Name-Swift.h
文件中,可以点进去查看。代码调用
1、swfit类要暴露给OC调用,这个类必须要继承自
NSObject
。
因为OC调用方法使用的是runtime
的消息机制,类需要有isa
指针,但是swift类是没有的,所以要继承自NSObject
基类,才会有isa
指针。2、swift类需要暴露给OC调用的成员和方法需要用
@objc
来修饰一下。
具体代码如下:
Person.swift文件代码:
class Person: NSObject {
@objc var age : Int = 0
@objc init(age: Int) {
self.age = age
}
}
如果觉得每个成员和方法都要用@objc来暴露太繁琐,也可以使用@objcMembers
来暴露所有的成员和方法。
@objcMembers class Person: NSObject {
var age : Int = 0
init(age: Int) {
self.age = age
}
}
OC文件调用swift代码:
#import "Test-swift.h"
Person *p = [[Person alloc] initWithAge:10];
NSLog(@"%ld", p.age);
-
修改暴露的成员名称、方法名以及类名
@objc
除了可以暴露成员和方法外,还可以修改它们暴露出去的名称,也可以修改暴露出去的类名。
swfit文件代码
@objc(JJPerson)
@objcMembers class Person: NSObject {
var age : Int = 0
init(age: Int) {
self.age = age
}
@objc(JJSay)
func say() {
print("hello")
}
}
OC调用swift代码
#import "JJ-swift.h"
JJPerson *p = [[JJPerson alloc] initWithAge:10];
[p JJSay];
OC文件中使用导入的swift第三方库
导入swift第三方库时,会自动创建一个不可见的文件 库名-Swift.h
例如:
导入IQKeyboardManagerSwift
三方库,会自动生成IQKeyboardManagerSwift-Swift.h
文件,在需要使用IQKeyboardManagerSwift
库的OC文件里#import "IQKeyboardManagerSwift-Swift.h"
就可以使用它了。
使用示例:[IQKeyboardManager shared].enable = YES;
二、swift调用OC
创建桥接文件
1、一般在OC项目中第一次创建swift文件或者在swift项目中第一次创建OC文件时xcode会弹出询问我们是否需要创建这个文件,点击确定即可自动创建一个。
2、默认名称为工程名-Bridging-Header.h
,我的工程名为Test
所以默认为Test-Bridging-Header.h
3、如果没有自动创建,那么手动创建一个工程名-Bridging-Header.h
文件也是可以的。
4、如果不想使用默认的名字,也可以修改名称,但是要在Build-Setting->Objective-C Bridging Header中将配置修改为想要的文件路径。在桥接文件中导入供swfit调用的OC文件
swift文件调用OC
三、思考
-
1、OC调用swift方法走的是runtime机制吗?
是!因为swift类继承自NSObject才能让OC使用,所以类也有isa指针,runtime消息机制就是通过isa指针来寻找方法。
OC调用swift的JJSay方法,断点查看汇编可以看到红框部分的注释,有调用objc_msgSend,这个就是OC运行时的消息发送机制。
-
2、swift调用OC方法走的是runtime机制吗?
是!因为OC方法在.m文件里编译,所以肯定是runtime机制。
swift调用OC的eat方法,断点查看汇编可以看到红框部分的注释,有调用objc_msgSend,这个就是OC运行时的消息发送机制。
-
3、swift类继承了NSObject,并且也用@objc或者@objcMembers暴露了方法,在swift文件里直接使用这个类的方法是走的runtime机制吗?
不会!
查看汇编可以看到,调用方法这里并没有objc_msgSend。
-
4、swift文件调用swift方法希望它使用runtime怎么做呢?
在方法前加上dynamic
关键字修饰,如:dynamic func say() {print("hello")}
查看汇编,可以看到调用say方法时的注释有objc_msgSend。