OC和Swift混编

最近在尝试oc和swift的混编,一是:swift是大势所趋,终究是要学习的。二是:项目目前架构是oc架构,全部重构为swift也不现实。下面是混编中遇到的一些问题和总结:
1.在oc项目中第一次创建swift文件的时候,会弹框提示是否要创建桥接文件。当然要创建了,否则还要自己动手创建和配置。教程网上很多,就不重复了。 2.oc的代码中需要调用swift的文件,需要导入隐式头文件:xxx-Swift.h,xxx-Swift.h在项目中是看不到的,但是确实是可以import的
//xxx是你的项目名字(一般是target的名字) #import "xxx-Swift.h" //然后代码如下:Ps:DYUserInfoSwiftVC是新建的swift的类。其实在oc的代码中引用swift的类,代码风格是和oc保持一致。swift的代码中引用oc的类,代码风格和swift保持一致。//oc中引用swift(DYUserInfoSwiftVC是swift的类) DYUserInfoSwiftVC *userSetSwift = [[DYUserInfoSwiftVC alloc] init]; [weakSelf pushController:userSetSwift]; //swift引用oc(DYSettingMesssageSetVC是oc的类) let messageSet:DYSettingMesssageSetVC = DYSettingMesssageSetVC.init() self.navigationController?.pushViewController(messageSet, animated: true)

3.swift中引用oc的类,需要在桥接文件 “项目名称-Bridging-Header.h”中 import 用到的oc文件,不需要在swift类中引用桥接文件。swift引用swift类也不需要导入头文件,因为编译器帮我们处理了,直接用就可以了。如图: OC和Swift混编
文章图片
536DD5F6-EAA3-4E1E-9D89-740FF155BBB0.png 4.相关配置信息: OC和Swift混编
文章图片
2AEF27D6-F27E-42A2-8C0B-736D55274FFA.png
OC和Swift混编
文章图片
71FAF796-676F-4CE6-AD27-F2C95BF5E86D.png 5.下面说下我的项目中遇到的问题:
(1)我们的项目是多target管理的,有多个pch文件,pch文件有一些宏定义,管理上架版本和企业版本,代码中有一些根据宏定义的判断。

OC和Swift混编
文章图片
EF87EB72-6194-4698-A01F-1360A965F49D.png
oc代码中直接if(DY_APPSTORE)就可以做相应的逻辑处理了,然而swift中却不行,必须if(DY_APPSTORE == 0)这种形式进行判断。
if (0 == DY_APPSTORE) { //代码 }

(2)oc中引用swifit时需要导入隐式头文件:xxx-Swift.h,但是如果项目是多个target的话,会生成多个xxx-Swift.h隐式文件。所以需要进行判断区分,否则会提示某个xxx-Swift.h找不到。处理方式如下:
新建一个Swift.h文件,代码如下。只需要在需要引入swift的代码中导入Swift.h文件即可。

OC和Swift混编
文章图片
8769B25A-2056-4BCB-8709-5F743F7C9097.png
(3)oc中经常用#pragma mark - 来添加注解,swift中需要用://MARK:-lazy这种形式。
(4)swift中的宏定义和oc的有区别。oc中可以宏定义一个方法,方便直接调用,swift不能这么写,如:
//oc #define DYGetColorFromHex(rgbValue) [UIColor colorWithRed:((float)((rgbValue & 0xFF0000) >> 16))/255.0 green:((float)((rgbValue & 0xFF00) >> 8))/255.0 blue:((float)(rgbValue & 0xFF))/255.0 alpha:1.0]

在swift如何定义宏定义呢,可以新建一个DYSwiftDefine.swift的文件,直接写相应的函数方法就可以了,代码如下:
import Foundation //颜色值 func UIColorFromRGB(rgbValue: UInt) -> UIColor { return UIColor( red: CGFloat((rgbValue & 0xFF0000) >> 16) / 255.0, green: CGFloat((rgbValue & 0x00FF00) >> 8) / 255.0, blue: CGFloat(rgbValue & 0x0000FF) / 255.0, alpha: CGFloat(1.0) ) }

因为新建的DYSwiftDefine.swift文件不需要在swift文件中导入的,是可以直接在代码中调用的:
self.view.backgroundColor = UIColorFromRGB(rgbValue: 0xffffff);

(5)oc中调用swift方法和传值时遇到的坑:
上面讲到在oc中调用swift,只需要导入隐式头文件:xxx-Swift.h即可。这种方式是可以创建controller然后进行push操作,但是如果想调用swift类中的方法或者给block赋值就会报错。处理方式如下:
import UIKit typealias DYMusicModelBlock = (_ model:DYMusicListModelSwift)->Void @objcMembers class DYMusicListVCSwift: BaseViewController,UITableViewDelegate,UITableViewDataSource,DevPulldownRefreshDelegate,LSLoadMoreViewDelegate,DYInputOfSearchViewDelegate { //MARK:全局变量 public var musicModelBlock:DYMusicModelBlock? }

【OC和Swift混编】是不是注意到了“@objcMembers”
@objcMembers 在Swift 4中继承 NSObject 的 swift class 不再默认全部 bridge 到 OC,如果我们想要使用的话我们就需要在class前面加上@objcMembers 这么一个关键字。
然后就可以复制和使用相关的方法了
DYMusicListVCSwift *music = [[DYMusicListVCSwift alloc] init]; music.musicModelBlock = ^(DYMusicListModelSwift * model) { weakSelf.musicArray = [NSMutableArray arrayWithObjects:DevNSStringFilterEmptyString(model.musicIcon), DevNSStringFilterEmptyString(model.musicName), DevNSStringFilterEmptyString(model.musicUrl), DevNSStringFilterEmptyString(model.musicTime), DevNSStringFilterEmptyString(model.musicAuthor), DevNSStringFilterEmptyString(model.musicId), nil]; weakSelf.actualValueTextField.text = model.musicName; }; [self.navigationController pushViewController:music animated:YES];

(6) swift中的代码强转:有时候我们需要对一些id类型的对象进行强转成我们需要的对象。
//假设需要解析一个请求回调的block的数据 self.musicListSearchRequest.successBlock = {(response) in if let resultCode = responseDic["resultCode"] as? String{ //是string类型 }else{ //不是string类型,可能是服务器返回的数据类型和约定的不一样,或者你不能用String强转,因为返回的是Int类型。 }

主要就是 if let xxxx = xxxx["xxx"] as? xxxx{
}else{
}
做一个类型转换的判断。如果不判断,直接强转后直接用的话有可能crash,比如你认为强转后是String类型,其实是个Int类型。你调用string的方法isEmpty()的时候就crash。

    推荐阅读