Swift: 语法秘笈
wwlw5827
8年前
<p>作者:Andyy Hope, <a href="/misc/goto?guid=4959676864214979231" rel="nofollow,noindex">原文链接</a> ,原文日期:2016/07/20</p> <p>译者: <a href="/misc/goto?guid=4959676864294693954" rel="nofollow,noindex">冬瓜</a> ;校对: <a href="/misc/goto?guid=4959676864374654319" rel="nofollow,noindex">Darren</a> ;定稿:千叶知风</p> <p><img src="https://simg.open-open.com/show/054a127b57ab75f1e19a9243e43cba3c.png"></p> <p>↑ ↑ ↓ ↓ ← → ← → B A</p> <p>无论 Swift 是你的第一门开发语言,还是从 Objective-C 转来,Swift 这门强大的语言都值得我们去学习和使用,但是 Swift 语法的不断迭代更新可能会令你畏惧。本文将会列举一些常见的语法,来帮助你提高 Swift 语言能力,精炼代码。</p> <h2>闭包(Closure)</h2> <pre> <code class="language-swift">() -> Void </code></pre> <p>有些文章中也会称作 <em>匿名函数</em> (unnamed functions),类似于 C 或 Objective-C 中的 <em>block</em> ;闭包是一个很轻量但是功能十分强大的函数,常用于类间的值传递,闭包通常作为函数的参数来使用,当然也可以作为变量。</p> <p>如果你有过 iOS 开发经验,你在使用 <strong>UIView</strong> animation 的 API 时肯定会用到闭包:</p> <pre> <code class="language-swift">class func animateWithDuration(_ duration: NSTimeInterval, animations: () -> Void) </code></pre> <p>animation 参数:传入动画相关代码,例如:</p> <pre> <code class="language-swift">UIView.animateWithDuration(10.0, animations: { button.alpha = 0 }) </code></pre> <p>animationWithDuration 这个函数利用了闭包,最终我们看到的效果是 <strong>button</strong> 逐渐消失,直到 <strong>alpha</strong> 属性为 <strong>0</strong> (不可见状态)。</p> <h2>尾部闭包(Trailing closures)</h2> <pre> <code class="language-swift">UIView.animateWithDuration(10.0) { button.alpha = 0 } </code></pre> <p>Swift 的这个特点可以省去很多无用代码。我们再看上面的代码,仔细的同学已经发现在相同的 API 我们上面的写法节省了很多代码。</p> <p>因为在 animateWithDuration 方法中最后一个参数是闭包,顾名思义,称之为 <em>尾部闭包</em> 。尾部闭包允许我们省略参数名,并且能放置在参数表括号以外,进一步简洁代码。以下两个代码实现功能相同,但是后者使用了尾部闭包:</p> <pre> <code class="language-swift">func say(message: String, completion: () -> Void) { print(message) completion() } ... say("Hello", completion: { // prints: "Hello" // Do some other stuff }) say("Hello") { // prints: "Hello" // Do some other stuff } </code></pre> <h2>类型别名(Type Alias)</h2> <pre> <code class="language-swift">typealias </code></pre> <p>当我们大量的使用某一种类型来定义时,类型别名是一种方便的手段。比如说我们有一个函数,它的参数是闭包:</p> <pre> <code class="language-swift">func dance(do: (Int, String, Double) -> (Int, String, Double)) { } </code></pre> <p>这看上去并不复杂,但是如果我们想在多个函数间相互传递这个闭包呢?我们不得不记住他的参数名,以确保它在函数中可传递。如果参数名不相同,就无法编译成功,错误日志会提示在这个传递过程中保证参数名相同。</p> <pre> <code class="language-swift">func dance(do: (Int, String, Double) -> (Int, String, Double)) { } func sing(do: (Int, String, Double) -> (Int, String, Double)) { } func act(do: (Int, String, Double) -> (Int, String, Double)) { } </code></pre> <p>倘若我们交换参数顺序、改变返回值类型,同样的会出现上述问题。所以,一旦我们更改需求,我们需要更新所有出现这个闭包的地方,这种问题的处理方法将会十分繁琐。所以,我们引入 <em>类型别名</em> 来解决这个问题。</p> <pre> <code class="language-swift">typealias TripleThreat = (Int, String, Double) -> (Int, String, Double) ... func dance(dance: TripleThreat) { } func act(act: TripleThreat) { } func sing(sing: TripleThreat) { } </code></pre> <p>现在我们重写之前的所有方法。如果再想更改参数闭包的话,我们所要做的仅仅是修改 typealias 即可。</p> <h3>类型别名代表性的用法(Famous Type Aliases)</h3> <pre> <code class="language-swift">typealias Void = () typealias NSTimeInterval = Double </code></pre> <h2>参数名缩写(Shorthand argument names)</h2> <pre> <code class="language-swift">$0, $1, $2... </code></pre> <p>如果一个闭包中有一个或多个参数,Swift 允许我们通过参数名来访问参数:</p> <pre> <code class="language-swift">func say(message: String, completion: (goodbye: String) -> Void) { print(message) completion(goodbye: "Goodbye") } ... say("Hi") { (goodbye: String) -> Void in print(goodbye) } // prints: "Hi" // prints: "Goodbye" </code></pre> <p>这个例子中,我们的尾部闭包中有一个名为 goodbye 的 String 型参数,Xcode 会将这个参数放到一个元祖中,然后紧跟着一个类型代表返回值,最后再加上 in 关键字来代表参数的结束。下一行则是我们闭包的具体实现。当我们的闭包短小,具有高可读性时,我们可以追求更加简洁的写法。我们现在开始减少代码,以达到极小:</p> <pre> <code class="language-swift">(goodbye: String) -> Void in </code></pre> <p>很多代码都不是必要的,因为我们可以使用 <strong>参数名缩写</strong> 。</p> <pre> <code class="language-swift">say("Hi") { print($0) } // prints: "Hi" // prints: "Goodbye" </code></pre> <p>正如所见,可以省略 goodbye 的参数名,以及 Void 返回值。并且 in 关键字也可省略,因为我们没有使用到参数名称。由于简单,每个参数都会依照在闭包中的声明的顺序。甚至,我们可以将闭包做缩行处理。</p> <p>如果闭包中有多个参数,参数名缩写将会依照顺序排列,例如:</p> <pre> <code class="language-swift">(goodbye: String, name: String, age: Int) -> Void in // $0: goodbye // $1: name // $2: age </code></pre> <h3>Return Self</h3> <pre> <code class="language-swift">-> Self </code></pre> <p>Swift 2.0 发布的时候,带来了一系列的新特性例如 map 、 flatMap 等等。更有趣的是,在这些方法中,我们同样的可以使用 <em>$</em> 符号来通过序号对其操作:</p> <pre> <code class="language-swift">[1, 2, 3, nil, 5] .flatMap { $0 } // remove nils .filter { $0 < 3 } // filter numbers that are greater than 2 .map { $0 * 100 } // multiply each value by 100 // [100, 200] </code></pre> <p>很酷吧?这种写法比较优雅、可读,易于理解。我们应该在更多的地方使用它。</p> <p>另外,我们可以通过闭包创建一个 String 的扩展,我们对 String 上执行一堆操作,并返回自己而不是使函数返回无效:</p> <pre> <code class="language-swift">// extension UIView func withBackgroundColor(color: UIColor) -> Self { backgroundColor = color return self } func withCornerRadius(radius: CGFloat) -> Self { layer.cornerRadius = 3 return self } ... let view = UIView(frame: CGRect(x: 0, y: 0, width: 10, height: 10)) .withBackgroundColor(.blackColor()) .withCornerRadius(3) </code></pre> <h3>总结</h3> <p>无论你是在写新的功能还是在读旧的代码,你会发现这种精炼代码的方式在任何地方都适用,并且你已经掌握了精炼方法。由于 Xcode 的自动补全现在还不完善,所以你应该不断地去质疑自己的代码,不要过度的依赖自动补全,而是自主完成代码。</p> <p>另外,我在 <a href="/misc/goto?guid=4959676864458655298" rel="nofollow,noindex">github</a> 上提供了一个playground文件以方便你来测试以上内容。</p> <p>译者注:译者自己整理了原作者的示例代码,并加上中文注释,详见译者的 <a href="/misc/goto?guid=4959676864532224703" rel="nofollow,noindex">Github仓库</a></p> <p>如果你喜欢这篇文章,并且对你的Coding Style有所帮助,可以在 <a href="/misc/goto?guid=4959676864629484018" rel="nofollow,noindex">推ter</a> 联系并follow我。</p> <p>我将会在九月与一群swift爱好者参与 <a href="/misc/goto?guid=4959676864701364429" rel="nofollow,noindex"> <em>try! Swift NYC</em> </a> ,到时候我们不见不散。</p> <p>本文由 SwiftGG 翻译组翻译,已经获得作者翻译授权,最新文章请访问http://swift.gg。</p> <p> </p> <p>来自:http://swift.gg/2016/08/17/developing-tvos-apps-for-apple-tv-with-swift/</p> <p> </p> <p><span style="background:rgb(189, 8, 28) url("data:image/svg+xml; border-radius:2px; border:medium none; color:rgb(255, 255, 255); cursor:pointer; display:none; font:bold 11px/20px "Helvetica Neue",Helvetica,sans-serif; opacity:1; padding:0px 4px 0px 0px; position:absolute; text-align:center; text-indent:20px; width:auto; z-index:8675309">Save</span></p>