美文网首页
Swift 版本更新学习笔记

Swift 版本更新学习笔记

作者: 董艺璇 | 来源:发表于2016-04-27 16:14 被阅读356次

    Swift 1.2

    • 新增 set 数据结构
    • if let 嵌套,支持 if let a=b,c=d
    • let 常量的赋值可以在后面使用时才赋值
    • 类型转换使用!符号

    Swift 2.0

    try/catch

    在方法中加入 throws 关键词告诉 Swift 这个方法可能会失败

    enum MyError : ErrorType{
        case UserError
        case NetworkErrof
        case DiscoveryError
    }
    func doStuff() throws -> String {
        print("Do stuff 1")
        print("Do stuff 2")
        throw MyError.NetworkError
    
        return "Some return value"
    }
    
    do {
        try doStuff()
        print("Success")
    }catch{
        print("An error occured")
    }
    
    do {
        try doStuff()
        print("Success")
    }catch MyError.NetworkError{
        print("Network error occured")
    }catch{
        print("An error occured")
    }
    

    在这些代码里永远执行不到 doStuff 方法的 return 方法惹。

    automatically synthesized headers

    对于一个很大的 Swift 文件,查看这里面所有的方法是很麻烦的,Xcode 提供了一个简单有效的方法,Xcode 可以显示出合成的头文件,在里面概括出了这个文件中的方法。

    guard

    func submitTapped() {
        guard username.text.characters.count > 0 else {
            return
        }
        print("All good")
    }
    

    在方法前面加入条件检测。
    看起来使用 guard 关键词与实用 if 条件判断没啥区别,但是使用了 guard 关键词以后,你的意图更加明确:如果条件不满足,方法不会继续执。并且使用了 guard 关键词以后更加简洁易懂。
    另外一个 guard 的优点在于:实用它来 unwrap optional 值,这些值可以在后续代码中使用。

    guard let unwrappedName = userName else {
        return
    }
    print("Your username is \(unwrappedName)")
    
    if let unwrappedName = userName {
        print("Your username is \(unwrappedName)")
    } else {
        return
    }
    // this won't work – unwrappedName doesn't exist here!
    print("Your username is \(unwrappedName)")
    

    measuring strings has changed

    计算字符串中字符数使用的方法为 testString.characters.count

    defer

    在一些语言中使用 try/finally 关键词告诉你的程序:无论发生了什么,finally 中的代码都要执行。Swift 2 引入了 defer 关键词:它表明我希望这一段代码发生,但不是现在。通常意义来说,这意味着这段代码在方法结束前执行。并且就算抛出了一个error,这段代码也会执行。

    一个简单的示例:

    override func viewDidLoad() {
        super.viewDidLoad()
        print("Checkpoint 1")
        doStuff()
        print("Checkpoint 4")
    }
    func doStuff() {
        print("Checkpoint 2")
        defer { print("Do clean up here") }
        print("Checkpoint 3")
    }
    

    这段代码的输出顺序为:

    Checkpoint 1
    Checkpoint 2
    Checkpoint 3
    Do clean up here
    Checkpoint 4
    

    一个不辣么简单的示例:

    override func viewDidLoad() {
        super.viewDidLoad()
        print("Checkpoint 1")
        do {
            try doStuff()
        } catch {
            print("Error!")
        }
        print("Checkpoint 4")
    }
    func doStuff() throws {
        print("Checkpoint 2")
        defer { print("Do clean up here") }
        throw MyError.UserError
        print("Checkpoint 3")
    }
    

    这段代码的输出为:

    Checkpoint 1
    Checkpoint 2
    Do clean up here
    Error
    Checkpoint 4
    

    mutability warnings

    如果在 Swift 中定义了变量但是从未改变它的值,Xcode 会给你警告。

    checking API availability

    if #available(iOS 9, *) {
        let stackView = UIStackView()
        // do stuff
    }else{
        //do other stuff
    }
    

    #available: 会自动监测是否运行在 iOS 9 及以上的环境,如果是,则继续运行这段代码,否则做其他处理。

    Swift 2.1

    Swift 2.2

    编译时 Swift 版本监测

    #if swift(>=3.0)
    print("Running Swift 3.0 or later")
    #else
    print("Running Swift 2.2 or earlier")
    #endif
    

    编译时监测 selectors

    Swift 2.2 反对使用 String 作为 selectors ,并且引入了新的语法:#selector
    使用 #selector 会在编译时检查你的代码来保证要调用的方法却是存在,否则编译会报错,从而无法build你的应用。从而避免了 String 作为 selector 时名称错误情况的发生。

    更多的作为参数标签的关键词

    Swift 有很多关键词:例如 class , func , let ,以及 public 有着特殊的意义并且不能被用作标识符。 Swift 允许你使用关键词作为参数标签,但是需要把它们放在引号内使用。

    示例:

    func visitCity(name: String, `in` state: String) {
        print("I'm going to visit \(name) in \(state)")
    }
    visitCity("Nashville", `in`: "Tennessee")
    

    在 Swift 2.2 以后,任何关键词(除了 inout , varlet )都可以被用作参数标签。不需要再把关键词放在引号中了。

    示例:

    func visitCity(name: String, in state: String) {
        print("I'm going to visit \(name) in \(state)")
    }
    visitCity("Nashville", in: "Tennessee")
    

    内嵌元组比较

    let singer = (first: "Taylor", last: "Swift")
    let alien = (first: "Justin", last: "Bieber")
    if singer == alien {
        print("They match! That explains why you never see them together…")
    } else {
        print("No match.")
    }
    

    输出结果为 No match

    let singer = (first: "Taylor", last: "Swift")
    let bird = (name: "Taylor", breed: "Swift")
    if singer == bird {
        print("This explains why she sings so well.")
    } else {
        print("No match.")
    }
    

    输出结果为: his explains why she sings so well.
    注意点:Swift2.2 忽略了元素名称,所以 singerbird 相等。

    反对元组的splat(拍打?)语法,很不常见略

    反对C-风格的循环

    C-风格的循环示例:

    for var i = 0; i < 10; i++ {
        print(i)
    }
    for var i = 10; i > 0; i-- {
        print(i)
    }
    
    for var i = 0; i < 10; i += 2 {
        print(i)
    }
    

    推荐写法:

    for i in 0 ..< 10 {
        print(i)
    }
    for i in (1...10).reverse(){
        print(i)
    }
    for i in 0.stride(to: 10, by: 2){
        print(i)
    }
    

    反对使用++和--

    i ++ 变成 i += 1
    i -- 变成 i -= 1
    

    反对使用var参数

    示例:

    func greet(var name: String) {
        name = name.uppercaseString
        print("Hello, \(name)!")
    }
    
    var name = "Taylor"
    greet(name)
    print("After function, name is \(name)")
    

    这个方法会带来这样的困惑:输出的名字是大写的还是原来的呢。
    在这种情况下推荐使用 inout 来修饰变量,在调用该方法传参时加上 & 符号

    func greet(inout name: String) {
        name = name.uppercaseString
        print("Hello, \(name)!")
    }
    var name = "Taylor"
    greet(&name)
    

    为了避免这类困惑产生,可以在函数中直接创建一个copy

    print("After function, name is \(name)")
    func greet(name: String) {
        let uppercaseName = name.uppercaseString
        print("Hello, \(uppercaseName)!")
    }
    

    renamed debug identifiers

    相关文章

      网友评论

          本文标题:Swift 版本更新学习笔记

          本文链接:https://www.haomeiwen.com/subject/dopcrttx.html