一、头文件路径
Xcode已经默认生成一个用于OC调用Swift的头文件,文件名格式是:
工程名称-Swift.h
(这个文件是存在的,只不过你看不到,需要用的时候直接导入头文件就可以了)-
设置文件路径
- TARGETS
- Build Swift Complier
- Objective-C Generated Interface Header Name
二、Swift的类暴露给OC的条件:
- 想要在OC调?Swift必须导?头?件
{targetName}-Swift.h
- Swift暴露给OC的类最终继承自NSObject
- 使用
@objc
修饰需要暴露给OC的成员- 使用
@objcMembers
修饰类,代表默认所有成员都会暴露给OC(包括扩展中定义的成员),最终是否成功暴露,还需要考虑成员自身的访问级别
- Swift类文件代码:
import Foundation
@objcMembers class Car: NSObject {
var price: Double
var band: String
init(price: Double, band: String) {
self.price = price
self.band = band
}
func run() { print(price, band, "run") }
static func run() { print("Car run") }
}
extension Car {
func test() { print(price, band, "test")}
}
按住「command」键选中#import "SwiftDemo-Swift.h"
文件可查看Swift代码生成对应的OC声明
#import "SwiftDemo-Swift.h"
@interface Car : NSObject
@property (nonatomic) double price;
@property (nonatomic, copy) NSString * _Nonnull band;
- (nonnull instancetype)initWithPrice:(double)price band:(NSString * _Nonnull)band OBJC_DESIGNATED_INITIALIZER;
- (void)run;
+ (void)run;
- (nonnull instancetype)init SWIFT_UNAVAILABLE;
+ (nonnull instancetype)new SWIFT_UNAVAILABLE_MSG("-init is unavailable");
@end
@interface Car (SWIFT_EXTENSION(SwiftDemo))
- (void)test;
@end
- 调用
Car *car = [[Car alloc] initWithPrice:10.5 band:@"BMW"];
car.band = @"Bently";
car.price = 108.5;
[car run]; // 108.5 Bently run
[car test]; // 108.5 Bently test
[Car run]; // Car run
@objc
可以通过@objc
重命名Swift暴露给OC的类名、属性名、函数名等
- Swift代码
@objc(MyCar) //修改类名
@objcMembers class Car: NSObject {
var price: Double
@objc(name) //修改属性名
var band: String
init(price: Double, band: String) {
self.price = price
self.band = band
}
@objc(drive) //修改方法名
func run() {
print(price, band, "run")
}
static func run() {
print("Car run")
}
}
extension Car {
@objc(exec:v2:) //修改方法名
func test() {
print(price, band, "test")
}
}
- 在OC中调用
MyCar *c = [[MyCar alloc] initWithPrice:10.5 band:@"BMW"];
c.name = @"Bently";
c.price = 108.5;
[c drive]; // 108.5 Bently run
[c exec:10 v2:20]; // 108.5 Bently test
[MyCar run]; // Car run
思考:
1.为什么Swift暴露给OC的类最终要继承?NSObject?
- 因为如果想要?OC的那?套,必须要使?isa,必须要?Runtime那?套,所以必须要继承于NSObject,因为NSObject有isa,纯Swift调?还是?虚表那?套。
2.Swift调?OC,底层是怎么调?的?OC调?Swift底层?是如何调??
- 例如:
p.run()
?的还是OC的objc_msgSend和Runtime那?套。 - 也是?的OC的objc_msgSend和Runtime那?套(因为Swift暴露给OC的必须要继承NSObject)。
3.如果Swift暴露给OC的类,但是还在Swift中调?,那么?的还是Swift虚表那?套,并不是?OC,因为还在Swift中调?,没必要?OC(如果真的想走OC的Runtime那?套,可以加dynamic关键字)。
注意:
-
Swift的类内存结构是:
- 前8个字节放元类型(metadata,元数据)相关,
- 后8个字节放指针相关,后?再放成员变量信息。
- 如果继承于NSObject,内存信息就变成了:前8个字节放isa指针相关,后?再放成员变量信息。