Swift与OC的混编可以总结出以下几种场景
OC&Swift Mixed In The Same Target Project中OC 调用 SwiftPod中OC 调用 SwiftProject中Swift 调用 OCPod中Swift 调用 OC OC&Swift Mixed In The Different Target OC Project 调用 Swift PodSwift Project 调用 OC PodOC Pod 调用 Swift PodSwift Pod 调用 OC Pod这里把Pod替换为Framework也是一样的,有时也把Project Target说成APP Target。1是同一target内混编,2是target间混编。
主工程里OC&Swift相互调用
在OC工程里创建一个Swift类会跳出如下d框
点击Create Bridging Header按钮生成桥接文件MainProject-Bridging-Header.h和MainProject-Swift, MainProject是工程名称。同时会在Build Settings里添加Swift Complier配置:
如果没有点击Create Bridging Header按钮,而是点击了Don’t create,那么将不会创建桥接文件,但是Build Settings里添加Swift Complier配置项,只是该配置项里没有Bridging Header,这时需要开发者自行创建桥接文件,并配置好路径。
OC调用Swift接口
Swift类里的接口应该加上@objc修饰
class Dog: NSObject {
@objc let legNumber = 4
var temper = "temper-good"
@objc var friend = Cat()
@objc func eat() {
print("The dog is eating")
}
}
在OC类中首先要引用MainProject-Swift.h
#import "MainProject-Swift.h"
最后就可以在OC类里引用Swift类
#import "MainProject-Swift.h"
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
Dog *dog = [[Dog alloc] init];
[dog eat];
}
@end
结论
我们再看看MainProject-Swift.h到底是什么。MainProject-Swift.h是一个自动生成的头文件,工程里没有对应的实体文件,可以Command+LeftClick点击MainProject-Swift.h进去看到其中有一段
@class Cat;
SWIFT_CLASS("_TtC11MainProject3Dog")
@interface Dog : NSObject
@property (nonatomic, readonly) NSInteger legNumber; // @objc let legNumber = 4
@property (nonatomic, strong, getter=friend, setter=setFriend:) Cat * _Nonnull friend_; //@objc var friend = Cat()
- (void)eat; //@objc func eat()
- (nonnull instancetype)init OBJC_DESIGNATED_INITIALIZER;
@end
在Cat这个Swift类中加了@objc修饰的属性和方法都在MainProject-Swift.h中自动生成了对应的OC属性和方法,所以通过引用MainProject-Swift.h就可以去调用这些接口了。
Swift调用OC接口
Swift调用OC类略有不同
在MainProject-Bridging-Header.h桥接文件里引入需要被Swift调用的类
#import "Cat.h"
Swift可以直接使用MainProject-Bridging-Header.h里引入的类
class Dog: NSObject {
@objc var friend = Cat()
func friendEatAction() {
friend.eat();
}
}
这个桥接过程比较简单。
如果主工是Swift编写的
这种情况下OC&Swift的相互调用与上述并无二致。在创建OC类的时候需要同时生成一个桥接文件和一个Swift转OC的接口文件,其他过程就不再赘述了。
OC主工程里调用Swift Pod
首先要确保OC主工程处于混编模式,否则再引入swfit pod,执行pod install时会报错(这个问题在问题2中有说明),此时最好是新建一个swift文件,自动创建桥接文件。
创建Swift类和接口。由于Swift接口是OC类调用的,所以Swift接口要有@objc修饰符; 另外接口调用是跨Target的,所以要有public或open关键字。如下代码,class前面和接口前面有@objc public修饰的就可以在主工程里使用。
@objc public class Dog: NSObject {
@objc let legNumber = 4
var temper = "temper-good"
@objc public func eat() {
print("The dog is eating")
}
}
OC调用Swift接口, Swift Pod工程的build setting里也有一个Interface Header文件,首先要在主工程里引入这个文件,让后就可以在OC里面使用Swift相关接口了。(除此之外还有@import导入方式,这是module机制)
#import "ViewController.h"
#import <SwiftPodFirst/SwiftPodFirst-Swift.h>
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
Dog *dog = [[Dog alloc] init];
[dog eat];
}
如果调用没有public的接口就会产生如下报错,OC可以访问的接口可以进入接口头文件SwiftPodFirst-Swift.h查看。
No visible @interface for 'Dog' declares the selector 'temper'
Swift主工程里调用OC Pod
首先要确保Swift主工程处于混编模式,具体方法不再赘述。
在Swift主工程桥接文件SwiftMainProject-Bridging-Header.h中添加#import引用
#import <OCPodSecond/Cat.h> // 这里OCPodSecond是OC pod,Cat是OC类
Swift主工程中使用OC类
import UIKit
class ViewController: UIViewController {
let cat = Cat() // Cat是OC类
override func viewDidLoad() {
super.viewDidLoad()
cat.eat() // 是OC对象的方法
}
}
一个由OC语言创建的Pod里有Swift代码,再由一个OC主工程使用这个Pod(这是项目里最常用到的场景,也是最复杂的一个场景)
在之前的OCPodSecond里添加一个Swift类,使OC Pod变成混编模式。
在第1步之后会自动创建OCPodSecond-Swift.h接口头文件,但并不会有创建OC-Bridging-Header.h的提示,那只好自己创建一个桥接文件并在build setting里关联路径
遵循之前的原则为Swift类添加@objc public接口
// 这个类是OCPodSecond里的Swift类
@objc public class Pig: NSObject {
@objc public func eat() {
print("The pig is eating!!")
}
}
最后在主工程里调用
#import "ViewController.h"
#import <SwiftPodFirst/SwiftPodFirst-Swift.h>
#import <OCPodSecond/Cat.h>
#import <OCPodSecond/OCPodSecond-Swift.h>
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
Dog *dog = [[Dog alloc] init]; //
[dog eat];
Cat *cat = [[Cat alloc] init]; //
[cat eat];
Pig *pig = [[Pig alloc] init]; // 在OC Pod里添加的Swift类 Pig
[pig eat];
}
运行一下试试,结果出现报错提示,framework targets不支持bridging header(引文7),
<unknown>:0: error: using bridging headers with framework targets is unsupported
Command CompileSwiftSources failed with a nonzero exit code
在Support Files里有个OCPodSecond-umbrella.h头文件,这个头文件替代了bridging header的作用,文件中会自动导入OC类。因此在Pod里混编更加简单了,桥接文件和相关接口会自动生成。
另外,每当新增一个接口时,都有可能出现接口不可用的情况,这或许是因为语言之间的转换没有这么快,也可能是存在一些接口缓存文件没有及时更新。遇到这种情况,最好是重新build一下或者把pod重新装一遍再或者clean重启一下工程。
OC Pod引用Swift Pod
这种情形与主工程调用Swift Pod没有区别,只需引入相关的接口文件就可以xxx-Swift.h,此处不再多述。需要注意的是,在podspec文件里要添加dependency,否则会无法引用相关pod。
Swift Pod 引用 OC Pod
这种情况有点特殊,虽然umbrella header有Bridging header的作用,但是umbrella header里的#import指令是自动生成的,并且只有Pod内部混编时才会自动引用Pod内的OC header。所以这样只能支持Pod内部混编。
假设一个Swift Pod需要引用一个OC Pod,此时在Swift Pod的umbrella header里添加#import “
这时我们需要modular的编译方式,在podfile里把use_frameworks!改为use_modular_headers!,这样就可以使用import 方式引用pod,import 方式同时兼容OC和Swift。
总结
最后混编路径汇总成下图,如果完全使用modular方式,可以把该图做一些简化。
问题1
[!] Unable to determine Swift version for the following pods:
- `SwiftPodFirst` does not specify a Swift version and none of the targets (`MainProject`) integrating
it have the `SWIFT_VERSION` attribute set. Please contact the author or set the `SWIFT_VERSION` attribute
in at least one of the targets that integrate this pod.
需要指定pod的swift版本,以便提供合适的编译器。修改方式是在集成的Target里指定SWIFT_VERSION。例如我们这里产生的原因是OC主工程不是混编模式,所以没有swift相关的编译项,这里修改的方式是在OC主工程里新建一个Swift类,让主工程变为混编模式,会自动在build setting生成相关的swift编译项。
问题2
Module 'SwiftPodFirst' not found
找不到某个Module,这是我们再开发中最常遇到的问题了,导致这个报错的原因很多。
更多详情查看引用链接
链接:https://www.jianshu.com/p/7e8312f3d8f1
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)