swift 可选型总结

Objective-C与Swift语言对于可选nil的不同理解:

Objective-C中的nil:表示缺少一个合法的对象,是指向不存在对象的指针,对结构体、枚举等类型不起作用(会返回NSNotFound)
Swift中的nil:表示任意类型的值缺失,是一个确定的值,要么是该类型的一个值要么什么都没有(即为nil)

一、申明可选常量或变量(非可选型不可以设置为nil,否则会报错)

let status: Int? = 1   // 申明可选Int类型的常量,初始值为1 
var defaultAddress: String? = "江苏南京"   // 申明可选String类型的变量,初始值为"江苏南京"
var student: Person? // 申明可选Person(自定义的类)的变量,初始值为nil

二、使用"!"强制解析获取可选类型的值(不建议直接使用)

var defaultAddress: String? = "江苏南京"
if defaultAddress != nil { // !=或==可以用来判断是否为nil
    print("您的地址是\(defaultAddress!)") // 使用!强制解析
} else {
    print("对不起,您不存在地址信息")
}

var student: Person?
print("学生为\(student!)") // XCode会提示运行错误,因为student初始值为nil,强制解析不行

三、使用可选绑定获取可选类型的值(建议的用法)

var defaultAddress: String? = "江苏南京"
if let address = defaultAddress { // 如果defaultAddress有值或类型转换成功,则将值赋值给address直接使用
    print("您的地址是\(address)")  // 使用address代替defaultAddress,且不需要加!强制解析
} else {
    print("对不起,您不存在地址信息")
}

四、隐式解析可选类型(用于申明时肯定有初始值,但后面可能为nil)

var mobileNumber: Int64! = 13912345678 // 第一次申明有初始值
//var mobileNumber: Int64! = nil // 这样也不会崩溃
print("您的电话号码是\(mobileNumber)") // 不需要使用!强制解析
// 打印内容:**您的电话号码是****Optional(13912345678)**
// 还是不建议直接强制解析,因为实际项目中可能中间已经对该值做了改变,若为nil则会运行错误导致APP崩溃
if let number = mobileNumber { // 建议的做法
    print("您的电话号码是\(number)")
// 打印内容:**您的电话号码是****13912345678**
} else {
    print("您没有记录电话号码")
}

五、空合运算符(用于判断变量或常量是否为nil)

// 空合运算符:a ?? b 判断a是否为nil,若a不为nil对a解封,否则返回b的值
var status: Int? // 申明可选Int类型的变量status,初始值为nil
status ?? 0 // 因为status为nil,则返回0
// ?? 即为以下if else的缩写
func testOption() -> Int {
    let status: Int? = 1
    if status == nil {
        return 0
    } else {
        return status!
    }
}

六、函数/方法返回类型为可选类型

A:返回值为可选类型的值(如Int?、String?、(Int, String)?、[Int]?、[Int: String]?等)
func returnPossibleValue(value: Bool) -> String? { // 返回类型为可选String类型
    if value {
        return "返回类型是可选类型值" // 如果为真,返回Int类型的值1
    } else {
        return nil // 如果为假,返回nil
    }
}

let possibleValue = returnPossibleValue(value: true) // 要用可选绑定判断再使用,因为possibleValue为String?可选类型
if let value = possibleValue {
    print(value)
} else {
    print("none value")
}
B:返回值为可选类型的类(如URL?、自定义Person?等)
class SomeClass {
    var someValue: Int
    init?(someValue: Int) { // 可失败构造器 
        if someValue == 1 { return nil }
        self.someValue = someValue
    }
}

func returnPossibleClass(value: Bool) -> SomeClass? { // 返回的类实例可能为nil
    if value {
        return SomeClass(someValue: 1) // 返回的为nil
    } else {
        return SomeClass(someValue: 2) // 返回的SomeClass?实例,不为nil
    }
}
C:返回值为可选类型的闭包(如(()-> (void))? )
func returnOptionalFunc(value: Bool) -> (() -> (Void))? { // 返回类型为可选类型的闭包
    if value {
        return { () in
            print("返回类型是可选类型闭包")
        }
    } else {
        return nil
    }
}

let possibleFunc = returnOptionalFunc(value: true) // 要用可选绑定判断再使用,因为possibleFunc 为可选类型的闭包,类型为() -> (Void)
if let aFunc = possibleFunc {
    print(aFunc())  // 注意增加()调用闭包,因为没有参数则是空括号
} else {
    print("none func")
}

七、可选类型在类或结构体中的运用

A:可选类型在类中的运用
class PossibleClass {
    var someValue: Int
    var possibleValue: String? // 可选存储属性,默认值为nil
    init(someValue: Int) { // 构造方法中可以不对possibleValue属性初始化
        self.someValue = someValue
    }
}

let someClass = PossibleClass(someValue: 4) // 实例化对象时不需要

注意:类中所有属性都需要有默认值。属性可以在申明时赋予初始值,也可以在构造方法中赋予初始值;子类继承父类时,必须先给自身属性先初始化后再继承父类构造方法初始化。一般的,出于安全的因素,子类的属性都赋予初始值或直接定义为可选类型。

B:可选类型在结构体中的运用

struct PossibleStruct {
    var someValue: Int
    var possibleValue: String? // 可选存储属性,默认值为nil
    // 结构体中可以自定义一个init构造器对属性初始化,也可以不自定义
}

let someStruct = PossibleStruct(someValue: 4, possibleValue: nil)

八、可选类型在构造器中的运用

概念:可失败构造器是 一个类、结构体或枚举类型的对象,在构造过程中有可能失败;这里所指的“失败”是指,如给构造器传入无效的参数值,或缺少某种所需的外部资源,又或是不满足某种必要的条件等。
A:结构体的可失败构造器
struct PossibleStructInit {
    let someValue: String
    init?(someValue: String) { // 可失败构造器
        if someValue.isEmpty { return nil } // 如果实例化为空串,则返回nil(即实例化失败)
        self.someValue = someValue
    }
}

let oneStruct = PossibleStructInit(someValue: "abc") // abc不是空串,oneStruct为PossibleStructInit?可选类型
if let one = oneStruct { // 使用if let可选绑定判断
    print(one.someValue)
} else {
    print("none value")
}
// 结果打印 abc

let twoStruct = PossibleStructInit(someValue: "") // 传参为空串
if let two = twoStruct {
    print(two.someValue)
} else {
    print("none value")
}
// 结果打印 none value
B:枚举的可失败构造器
enum PossibleEnmuInit {
    case East, West, South, North // 不带原始值,带原始值也可以使用可失败构造器
    init?(director: Character) {
        switch director {
        case "E":
            self = .East
        case "W":
            self = .West
        case "S":
            self = .South
        case "N":
            self = .North
        default:
            return nil // 如果实例化时不是E/W/S/N字符,则返回nil(即实例化失败)
        }
    }
}

let oneEnmu = PossibleEnmuInit(director: "E")  // 传值E,返回PossibleEnmuInit?可选类型
if let one = oneEnmu { // 使用可选绑定判断
    print(one)
} else {
    print("none value")
}
// 结果打印 East

let twoEnmu = PossibleEnmuInit(director: "A")  // 传值A,返回PossibleEnmuInit?可选类型
if let two = twoEnmu { // 使用可选绑定判断
    print(two)
} else {
    print("none value")
}
// 结果打印 none value
C:类的可失败构造器
class Product {
   let name: String
   init?(name: String) {   
       if name.isEmpty { return nil }
       self.name = name
   }
}

class CartItem: Product {
   let quantity: Int
   init?(name: String, quantity: Int) {
       if quantity < 1 { return nil } // 实例化时如果quantity小于1,则立即终止构造过程
       self.quantity = quantity
       super.init(name: name) // 如果name为空串,则立即终止构造过程
   }
}

if let twoSocks = CartItem(name: "sock", quantity: 2) {
   print("Item: \(twoSocks.name), quantity: \(twoSocks.quantity)")
}
// 打印 "Item: sock, quantity: 2”

if let zeroShirts = CartItem(name: "shirt", quantity: 0) {
   print("Item: \(zeroShirts.name), quantity: \(zeroShirts.quantity)")
} else {
   print("Unable to initialize zero shirts")
}
// 打印 "Unable to initialize zero shirts”

九、可选类型在可选链中的运用

概念:可选链为一种可以在当前值可能为nil的可选值上请求和调用属性、方法及下标的方法。如果可选值有值,那么调用就会成功;如果可选值是nil,那么调用将返回nil。多个调用可以连接在一起形成一个调用链,如果其中任何一个节点为nil,整个调用链都会失败,即返回nil。
class Person { 
    var name: String
    var room: Room? // 不是每个人都买房,定义为可选类型
    init(name: String) {
        self.name = name
    }
}

class Room {
    var roomAddr: String
    var roomArea: Int
    init(roomAddr: String, roomArea: Int) {
        self.roomAddr = roomAddr
        self.roomArea = roomArea
    }
}

let xiaoming = Person(name: "xiaoming") // 此时xiaoming实例中属性:name为"xiaoming",room为nil
if let address = xiaoming.room?.roomAddr, let area = xiaoming.room?.roomArea { // 使用?可选链式调用,如果含有值则返回该值(不需要手动强制解析),如果没有值则返回nil
    print("\(xiaoming.name)的房子地址在:\(address), 面积:\(area)")
} else {
    print("\(xiaoming.name)没有房子")
}
// 打印:xiaoming没有房子

xiaoming.room = Room(roomAddr: "南京雨花台区", roomArea: 95) // 定义xiaoming实例的room属性
// 再次调用以上可选绑定,打印 xiaoming的房子地址在:南京雨花台区, 面积:95

十、可选类型在错误处理中的运用

概念:错误处理即为响应错误一级从错误中恢复的过程

A:一般的错误处理写法(一般使用于需要对不同的返回结果较为清晰的捕捉并做相应的处理)
// 操作错误处理枚举
enum OperationError: Error {
    case ErrorOne
    case ErrorTwo(String) // 带关联值的枚举属性
    case ErrorOthers
}

// 错误处理抛出
func throwDriver(num: Int) throws {
    if num == 1 {
        throw OperationError.ErrorOne
    } else if num == 2 {
        throw OperationError.ErrorTwo("数据类型错误")
    } else {
        throw OperationError.ErrorOthers
    }
}

do {
    print("使用do-catch捕获错误")
    try throwDriver(num: 2) // 使用try尝试请求
    print("未捕获到错误")
} catch OperationError.ErrorOne { // catch去捕捉是否又throw出来的错误
    print("捕捉到错误:ErrorOne")
} catch OperationError.ErrorTwo(let message) { // 可以使用let 获取枚举关联值
    print("捕捉到错误:ErrorTwo" + message)
} catch OperationError.ErrorOthers {
    print("捕捉到错误:ErrorOthers")
}
// 打印结果如下:
// 使用do-catch捕获错误
// 捕捉到错误:ErrorTwo数据类型错误
B:错误处理(try!)(不建议使用,可能会导致App崩溃)
let photo = try! loadImage(atPath: "./Resources/John Appleseed.jpg")
// 上述语句中在执行loadImage方法时如果执行失败,使用try!来禁用错误传递,会有运行错误导致App崩溃
C:错误处理(try?)(一般使用于不需要对错误进行过多的处理)
func someThrowingFunction() throws -> Int {
    // ...
}

let x = try? someThrowingFunction() // x可能正常返回一个Int类型的值也有可能抛出一个错误异常,使用时对x用if let可选绑定判断

十一、可选类型在类型转换中的运用

类型转换在向上转型时一般不会出现失败,例如从子类向父类转型,直接用as即可

类型转换在向下转型时可能会出现失败,例如从父类向子类转型,要使用as?转型,使用时需要可选绑定后再用值得注意的是:从Any转成Int为向下转型,Swift从安全因素考虑,会直接返回nil,所以在日常项目中若遇到从父类向子类转型时,一定要使用可选绑定,如以下代码:

class SuperClass {
    // 这是一个父类
}

class SubClass: SuperClass {
    // 这是一个子类
}

let superClass = SuperClass()
if let someClass = superClass as? SubClass {
    // 如果转换成功,则用someClass使用即可
} else {
    // 转换失败
}

参考:https://www.jianshu.com/p/f2e25f78f5a2

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

推荐阅读更多精彩内容