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混编_第1张图片
536DD5F6-EAA3-4E1E-9D89-740FF155BBB0.png
4.相关配置信息:
OC和Swift混编_第2张图片
2AEF27D6-F27E-42A2-8C0B-736D55274FFA.png

OC和Swift混编_第3张图片
71FAF796-676F-4CE6-AD27-F2C95BF5E86D.png
5.下面说下我的项目中遇到的问题:

(1)我们的项目是多target管理的,有多个pch文件,pch文件有一些宏定义,管理上架版本和企业版本,代码中有一些根据宏定义的判断。


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混编_第4张图片
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?
}

是不是注意到了“@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。

你可能感兴趣的:(OC和Swift混编)