Swift优雅的进行解包

对于Swift学习而言,可选类型Optional是永远绕不过的坎,特别是从OC刚刚转Swift的时候,可能就会被代码行间的?与!,有的时候甚至是??搞得稀里糊涂的.

这篇文章会给各位带来我对于可选类型的一些认识以及如何进行解包,其中会涉及到Swift中if let以及guard let的使用以及思考,还有涉及OC部分的nullablenonnull两个关键字,以及一点点对两种语言的思考.

var num: Int?它是什么类型?

在进行解包前,我们先来理解一个概念,这样可能更有利于对于解包.

首先我们来看看这样一段代码:


var num: Int?

num = 10

if num is Optional<Int> {

 print("它是可选类型")

}else {

 print("它是Int类型")

}

请先暂时不要把这段代码复制到Xcode中,先自问自答,num是什么类型,是Int类型吗?

好了,你可以将这段代码复制到Xcode里去了,然后在IDE中的if上一定会出现这样一段话:


'is' test is always true

不是Int类,它是Optiona类型

那么Optional类型是啥呢--可选类型,具体Optional是啥,点进去看看你就知道了.在这里我就不多做解释了.

var num: Int?这是Optional的声明,意思不是"我声明了一个Optional的Int值",而是”我声明了一个Optional类型值,它可能包含一个Int值,也可能什么都不包含”,也就是说实际上我们声明的是Optional类型,而不是声明了一个Int类型!

以此类推String?是什么类型,T?是什么类型,答案各位心中已经明了吧.

正是因为num是一个可选类型,所以它才能赋值为nil, var num: Int = nil,这样是不可能赋值成功的,因为Int类型中没有nil这个概念!

这就是Swift与OC一个很大区别,在OC中我们的对象都可以赋值为nil,而在Swift中,能赋值为nil只有Optional类型!

解包的基本思路,使用if let或者guard let,而非强制解包

我们先来看一个简单的需求,虽然这个需求在实际开发中意义不太大:

我们需要从网络请求获取到的一个人的身高(cm为单位)以除以100倍,以获取m为单位的结果然后将其结果进行返回.

设计思路:

由于实际网络请求中,后台可能会返回我们的身高为空(即nil),所以在转模型的时候我们不能定义Float类型,而是定义Float?便于接受数据

如果身高为nil,那么nil除以100是没有意义的,在编译器中Float?除以100会直接报错,那么其返回值也应该为nil,所以函数的返回值也是Float?类型

那么函数应该设计成为这个样子是这样的:


func getHeight(_ height: Float?) -> Float?

如果一般解包的话,我们的函数实现大概会写成这样:


func getHeight(_ height: Float?) -> Float? {

 if height != nil {

 return height! / 100

 }

 return nil

}

使用!进行强制解包,然后进行运算

我想说的是使用强制解包固然没有错,不过如果在实际开发中这个height参数可能还要其他用途,那么是不是每使用一次都要进行强制解包?

强制解包是一种很危险的行为,一旦解包失败,就有崩溃的可能,也许你会说这不是有if判断,然而实际开发中,情况往往比想的复杂的多,所以安全的解包行为应该是通过if let 或者guard let来进行


func getHeight(_ height: Float?) -> Float? {

 if let unwrapedHeight = height {

 return unwrapedHeight / 100

 }

 return nil

}

或者


func getHeight(_ height: Float?) -> Float? {

 guard let unwrapedHeight = height else {

 return nil

 }

 return unwrapedHeight / 100

}

那么if let和guard let 你更倾向使用哪个呢?

在本例子中,其实感觉二者的差别不大,不过我个人更倾向于使用guard let.

原因如下:

在使用if let的时候其大括号类中的情况才是正常情况,而外部主体是非正常情况的返回的nil;

而在使用guard let的时候,guard let else中的大括号是异常情况,而外部主体返回的是正常情况.

对于一个以返回结果为目的的函数,函数主体展示正常返回值,而将异常抛出在判断中,这样不仅逻辑更清晰,而且更加易于代码阅读

解包深入

有这么一个需求,从本地路径获取一个json文件,最终将其转为字典,准备进行转模型操作

在这个过程中我们大概有这么几个步骤:

  1. 获取本地路径

func path(forResource name: String?, ofType ext: String?) -> String?

2. 将本地路径读取转为Data

init(contentsOf url: URL, options: Data.ReadingOptions = default) throws

3. JSON序列化

class func jsonObject(with data: Data, options opt: JSONSerialization.ReadingOptions = []) throws -> Any

4. 是否可以转为字典类型

我们可以看到以上几个函数中,获取路径获取返回的路径结果是一个可选类型,而转Data的方法是抛出异常,JSON序列化也是抛出异常,至于最后一步的类型强转是使用as! [Sting: Any]这样的操作

这个函数我是这来进行设计与步骤分解的:

函数的返回类型为可选类型,因为下面的4步中都有可能失败进而返回nil

虽然有人会说第一步获取本地路径,一定是本地有的才会进行读取操作,但是作为一个严谨操作,凡事和字符串打交道的书写都是有隐患的,所以我这里还是用了guard let进行守护

这个函数看起来很不简洁,每一个guard let 后面都跟着一个异常返回,甚至不如使用if let看着简洁

但是这么写的好处是:在调试过程中你可以明确的知道自己哪一步出错


func getDictFromLocal() -> [String: Any]? {

 //1 获取路径

 guard let path = Bundle.main.path(forResource: "test", ofType:"json") else {

 return nil

 }

 //2 获取json文件里面的内容

 guard let jsonData = try? Data.init(contentsOf: URL.init(fileURLWithPath: path)) else {

 return nil

 }

 //3 解析json内容

 guard let json = try? JSONSerialization.jsonObject(with: jsonData, options:[]) else {

 return nil

 }

 //4 将Any转为Dict

 guard let dict = json as? [String: Any] else {

 return nil

 }

 return dict

}

当然,如果你要追求简洁,这么写也未尝不可,一波流带走


func getDictFromLocal() -> [String: Any]? {

 guard let path = Bundle.main.path(forResource: "test", ofType:"json"),

 let jsonData = try? Data.init(contentsOf: URL.init(fileURLWithPath: path)),

 let json = try? JSONSerialization.jsonObject(with: jsonData, options:[]),

 let dict = json as? [String: Any] else {

 return nil

 }

 return dict

}

guard let与if let不仅可以判断一个值的解包,而是可以进行连续操作

像下面这种写法,更加最求的是结果,对于一般的调试与学习,多几个guard let进行拆分,也未尝不可

至于哪种用法更适合,因人而异

可选链的解包

至于可选链的解包是完全可以一步到位,假设我们有以下这个模型


class Person {

 var phone: Phone?

}

class Phone {

 var number: String?

}

Person类中有一个手机对象属性,手机类中有个手机号属性,现在我们有位小明同学,我们想知道他的手机号

小明他不一定有手机,可能有手机而手机并没有上手机号码


let xiaoming = Person()

guard let number = xiaoming.phone?.number else {

 return

}

这里只是抛砖引玉,更长的可选链也可以一步到位,而不必一层层进行判断,因为可选链中一旦有某个链为nil,那么就会返回nil

nullable和nonnull

我们先来看这两个函数,PHImageManager在OC与Swift中通过PHAsset实例获取图片的例子


[[PHImageManager defaultManager] requestImageForAsset:asset 

targetSize:size 

contentMode:PHImageContentModeDefault 

options:options 

resultHandler:^(UIImage * _Nullable result, NSDictionary * _Nullable info) {

 // 非空才进行操作 注意_Nullable,Swift中即为nil,注意判断

 if (result) {

 }

}];


PHImageManager.default().requestImage(for: asset, 

targetSize: size, 

contentMode: .default, 

options: options, 

resultHandler: { (result: UIImage?, info: [AnyHashable : Any]?) in

 guard let image = result else { return }

})

在Swift中闭包返回的是两个可选类型result: UIImage?与info: [AnyHashable : Any]?

而在OC中返回的类型是 UIImage * _Nullable result, NSDictionary * _Nullable info

注意观察OC中返回的类型UIImage * 后面使用了_Nullable来修饰,至于Nullable这个单词是什么意思,我想稍微有点英文基础的应该一看就懂--"可能为空",这不恰恰和Swift的可选类型呼应吗?

另外还有PHFetchResult遍历这个函数,我们再来看看在OC与Swift中的表达


PHFetchResult *fetchResult;

[fetchResult enumerateObjectsUsingBlock:^(id _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {

}];


let fetchResult: PHFetchResult

fetchResult.enumerateObjects({ (obj, index, stop) in

})

看见OC中Block中的回调使用了Nonnull来修饰,即不可能为空,不能为nil,一定有值,对于使用这样的字符修饰的对象,我们就不必为其做健壮性判断了.

这也就是nullable与nonnull两个关键字出现的原因吧--与Swift做桥接使用以及显式的提醒对象的状态

一点点Swift与OC的语言思考

我之前写过一篇文章,是说有关于一个字符串拼接函数的

从Swift来反思OC的语法

OC函数是这样的


- (NSString *)stringByAppendingString:(NSString *)aString;

Swift中函数是这样的


public mutating func append(_ other: String)

仅从API来看,OC的入参是很危险的,因为类型是NSString *

那么nil也可以传入其中,而传入nil的后果就是崩掉,我觉得对于这种传入参数为nil会崩掉的函数需要特别提醒一下,应该写成这样:


- (NSString *)stringByAppendingString:(NSString * _Nonnull)aString;

或者这样


- (NSString *)stringByAppendingString:(nonnull NSString *)aString;

以便告诉程序员,入参不能为空,不能为空,不能为空.重要的事情说三遍!!!

反观Swift就不会出现这种情况,other后面的类型为String,而不是String?,说明入参是一个非可选类型.

基于以上对于代码的严谨性,所以我才更喜欢使用Swift进行编程.

当然,Swift的严谨使得它失去部分的灵活性,OC在灵活性上比Swift卓越,但是从安全角度和编码的长远意义看Swift才是现在与未来.

最后想说的是Swift在国内的使用并不是很受拥戴,这点很无奈,因为和整个的大环境有关.

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,923评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,154评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,775评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,960评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,976评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,972评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,893评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,709评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,159评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,400评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,552评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,265评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,876评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,528评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,701评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,552评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,451评论 2 352

推荐阅读更多精彩内容