Swift3新特性

作者: FlyElephant | 来源:发表于2016-10-17 23:21 被阅读838次

    Swift3 改变了很多大量的内容,如果你的代码中不作出必要的改变肯定会被拒绝.如果你认为Swift从1.2和2.0的改变的是很大,那是因为你还没有看到3的改变.
    本文中会尽可能通过代码来讲解Swift3的重要变化,更新代码的时刻已经到来了.
    警告# 1:有很多的变化,其中一些看起来小。然而这些变化是一个一次性的事件,使语言更好地来,同时意味着在以后版本的变化是显着更小。

    所有的函数参数标签

    Swift2.0中的原有的函数和方法调用方式发生了彻底的改变.在Swift2.x及更早版本中,方法名称不需要设置第一个参数标签,现在调用方式必须显示显示设置第一个参数的标签,例如:
    <pre><code>names.indexOf("Taylor") "Taylor".writeToFile("filename", atomically: true, encoding: NSUTF8StringEncoding) SKAction.rotateByAngle(CGFloat(M_PI_2), duration: 10) UIFont.preferredFontForTextStyle(UIFontTextStyleSubheadline) override func numberOfSectionsInTableView(tableView: UITableView) -> Int func viewForZoomingInScrollView(scrollView: UIScrollView) -> UIView? NSTimer.scheduledTimerWithTimeInterval(0.35, target: self, selector: #selector(createEnemy), userInfo: nil, repeats: true)</code></pre>

    Swift3 要求所有的参数必须要设置标签,除非特别设置,通过方法名中的关于第一个参数的说明已经取去除.Swift2和Swift3对比:

    <pre><code>`
    names.indexOf("Taylor")
    names.index(of: "Taylor")

        "Taylor".writeToFile("filename", atomically: true, encoding: NSUTF8StringEncoding)
        "Taylor".write(toFile: "somefile", atomically: true, encoding: String.Encoding.utf8)
        
        SKAction.rotateByAngle(CGFloat(M_PI_2), duration: 10)
        SKAction.rotate(byAngle: CGFloat(M_PI_2), duration: 10)
        
        UIFont.preferredFontForTextStyle(UIFontTextStyleSubheadline)
        UIFont.preferredFont(forTextStyle: UIFontTextStyle.subheadline)
        
        override func numberOfSectionsInTableView(tableView: UITableView) -> Int
        override func numberOfSections(in tableView: UITableView) -> Int
        
        func viewForZoomingInScrollView(scrollView: UIScrollView) -> UIView?
        func viewForZooming(in scrollView: UIScrollView) -> UIView?
        
        NSTimer.scheduledTimerWithTimeInterval(0.35, target: self, selector: #selector(createEnemy), userInfo: nil, repeats: true)
        Timer.scheduledTimer(timeInterval: 0.35, target: self, selector: #selector(createEnemy), userInfo: nil, repeats: true)`</code></pre>
    

    回调中,NSTimer的调用方式发生的改变,我们可以在FileManager
    , Data, Date, URLRequest, UUID,NotificationCenter, 同样的我们会发现一些基础数据类型去除了“NS”前缀~

    关于第一个标签的设置会导致一些连锁反应,当使用其他的框架如UIKit,他们期待原有的 "no first parameter name"规则在Swift 3.
    这些是在Swift 2.2的签名:
    <pre><code>override func viewWillAppear(animated: Bool) override func tableView(tableView: UITableView, numberOfRowsInSection section: Int) -> Int override func didMoveToView(view: SKView) override func traitCollectionDidChange(previousTraitCollection: UITraitCollection?) func textFieldShouldReturn(textField: UITextField) -> Bool</code></pre>

    Swift3中,第一个参数标签,通过下划线来设置,如下:
    <pre><code>override func viewWillAppear(_ animated: Bool) override func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int override func didMoveToView(_ view: SKView) override func traitCollectionDidChange(_ previousTraitCollection: UITraitCollection?) func textFieldShouldReturn(_ textField: UITextField) -> Bool</code></pre>

    省略不必要的字符

    当Swift在2015年12月开放源代码的时候,新的API标准包含三个决定性的话:“省略不必要的字。其中的变化,会使代码调用更加简洁.
    Swift 2.2 例子:
    <pre><code>let blue = UIColor.blueColor() let min = numbers.minElement() attributedString.appendAttributedString(anotherString) names.insert("Jane", atIndex: 0) UIDevice.currentDevice()</code></pre>

    你能辨认出不必要的字符吗?当使用UIColor定义蓝色blue是必要的,blueColor是不必要的,Swift3中上面的代码:
    <pre><code>let blue = UIColor.blue let min = numbers.min() attributedString.append(anotherString) names.insert("Jane", at: 0) UIDevice.current</code></pre>
    对比之下,代码较之前更精简.
    Swift2.2 和 Swift3 代码中字符串对比如下:

    <pre><code>` " Hello ".stringByTrimmingCharactersInSet(.whitespaceAndNewlineCharacterSet())
    " Hello ".trimmingCharacters(in: .whitespacesAndNewlines)

        "Taylor".containsString("ayl")
        "Taylor".contains("ayl")
        
        "1,2,3,4,5".componentsSeparatedByString(",")
        "1,2,3,4,5".components(separatedBy: ",")
        
        myPath.stringByAppendingPathComponent("file.txt")
        myPath.appendingPathComponent("file.txt")
        
        "Hello, world".stringByReplacingOccurrencesOfString("Hello", withString: "Goodbye")
        "Hello, world".replacingOccurrences(of: "Hello", with: "Goodbye")
        
        "Hello, world".substringFromIndex(7)
        "Hello, world".substring(from: 7)
        
        "Hello, world".capitalizedString
        "Hello, world".capitalized
    

    `</code></pre>

    警告: capitalized始终是一个属性,但是lowercaseString和uppercaseString已经被转换成lowercased()和uppercased()替换.

    关于Swift的变化有很多,以上的对比并不是变化最大的,有些地方变化不是特别明显:
    <pre><code>dismiss(animated: true, completion: nil)</code></pre>
    第一眼看到dismiss时候,是不是想到 "dismiss what?" Swift 2.2中代码如下:
    <pre><code>dismissViewControllerAnimated(true, completion: nil)</code></pre>
    现在甚至是complete都是可选参数:
    <pre><code>dismiss(animated: true)</code></pre>
    同样的改变发生在 prepareForSegue() 方法中,现在如下:
    <pre><code>override func prepare(for segue: UIStoryboardSegue, sender: AnyObject?)</code></pre>

    枚举和属性中的大小字母替换成小写字母

    我们在使用类和结构体,枚举一直遵守着参数名称以大写字母开头,虽然大写字母与参数无关,Swift3中开始使用小写字母.Swift 2.2创建 NSURLRequest对象使用NSURLRequest(URL: someURL),注意大写字母"URL". Swift 3重写URLRequest(url: someURL),同时意味着将使用webView.request?.url?.absoluteString来读取webview的返回参数.

    当属性的名称的一部分是大写的时候就不是那么和谐,例如cgcolorcicolor,Swift3中的调用方式如下:
    <pre><code>let red = UIColor.red.cgColor</code></pre>

    这种变化有利于驱动一致性,所有的属性和参数都应该以小写字母开头,也没有例外.
    枚举同样在发生变化,从大写改变为小写,枚举是一个数据类型,意味着无论你使用的任何苹果的枚举都将小写:

    <pre><code>`UIInterfaceOrientationMask.Portrait // old
    UIInterfaceOrientationMask.portrait // new

        NSTextAlignment.Left // old
        NSTextAlignment.left // new
        
        SKBlendMode.Replace // old
        SKBlendMode.replace // new`</code></pre>
    

    这种微小的改变涉及到了可选数据类型,原有的可选类型在枚举中:
    <pre><code>enum Optional { case None case Some(Wrapped)}</code></pre>
    如果项目中原有用到了Some,需要切换到度对应的some,当然也可以通过不使用some,以下代码供参考:
    <pre><code>
    for case let .some(datum) in data {
    print(datum)
    }

        for case let datum? in data {
            print(datum)
        }`</code></pre>
    

    Swift形式的C函数导入

    Swift3中介绍了C函数的属性,允许开发者通过新的和简洁的方式导入C函数.例如,从“CGContext”映射属性的方式至一个CGContext对象.是的,这意味着原来类似CGContextSetFillColorWithColor()这种可怕的方法被移除.
    为了证明这一点,以下是Swift2.2中的例子:

    <pre><code>`
    let rectangle = CGRect(x: 0, y: 0, width: 512, height: 512)
    CGContextSetFillColorWithColor(ctx, UIColor.redColor().CGColor)
    CGContextSetStrokeColorWithColor(ctx, UIColor.blackColor().CGColor)
    CGContextSetLineWidth(ctx, 10)
    CGContextAddRect(ctx, rectangle)
    CGContextDrawPath(ctx, .FillStroke)

        UIGraphicsEndImageContext()`</code></pre>
    

    在Swift3中CGContex可以被视作一个对象,你可以调用方式,而不是重复的使用CGContext,所有代码重写如下:

    <pre><code>`
    if let ctx = UIGraphicsGetCurrentContext() {
    let rectangle = CGRect(x: 0, y: 0, width: 512, height: 512)
    ctx.setFillColor(UIColor.red.cgColor)
    ctx.setStrokeColor(UIColor.black.cgColor)
    ctx.setLineWidth(10)
    ctx.addRect(rectangle)
    ctx.drawPath(using: .fillStroke)

            UIGraphicsEndImageContext()
        }`</code></pre>
    

    提示:在Swift2.2和Swift3.0中UIGraphicsGetCurrentContext()返回了可空的CGContext,但是因为Swift3中方式的形式调用所以使用之前需要判空.

    C函数的映射无处不在,比如说CGPDFDocumentnumberOfPages属性,CGAffineTransform如果写起来也是非常惊人的,以下是一些新老代码对比:

    <pre><code>`
    CGAffineTransformIdentity
    CGAffineTransform.identity

        CGAffineTransformMakeScale(2, 2)
        CGAffineTransform(scaleX: 2, y: 2)
        
        CGAffineTransformMakeTranslation(128, 128)
        CGAffineTransform(translationX: 128, y: 128)
        
        CGAffineTransformMakeRotation(CGFloat(M_PI))
        CGAffineTransform(rotationAngle: CGFloat(M_PI))`</code></pre>
    

    动词和名词

    Swift3中的动词和名词让人比较坤混,下面是比较重要的Swift3 API指南,读取来有点绕:

    • "When the operation is naturally described by a verb, use the verb’s imperative for the mutating method and apply the “ed” or “ing” suffix to name its nonmutating counterpart"

    • "Prefer to name the nonmutating variant using the verb’s past participle"

    • "When adding “ed” is not grammatical because the verb has a direct object, name the nonmutating variant using the verb’s present participle"

    • "When the operation is naturally des

    Swift3相当于给我们上了一堂英语语法课,方法的改变很微妙,有时候会让人感到困惑.看一些简单的代码:

    <pre><code>`
    myArray.enumerate()
    myArray.enumerated()

        myArray.reverse()
        myArray.reversed()`</code></pre>
    

    Swift3中在每个方法的最后都加入一个“d”,返回一个值.
    当涉及到数组排序的时候这些规则有时候会导致混乱。Swift2.2中用sort()返回排序后的数组,并sortinplace()到位数组排序。Swift3,sort()更名sorted()(上述例子),和sortinplace()更名sort()。

    Swift3的这些变化很容易阅读,其中一些是微小的,但是会引入大量的破坏,苹果让工程师的生活更加的建安,同时也会促进工程师的技能提升.

    [https://www.hackingwithswift.com/swift3]

    相关文章

      网友评论

        本文标题:Swift3新特性

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