1. 程式人生 > >iOS開發系列--Swift 3.0

iOS開發系列--Swift 3.0

概述

從寫第一篇Swift文章的時候到現在Swift已經從1.2發展到了今天的3.0,這期間由於Swift目前還在發展階段並不能向下相容,因此第一篇文章中的部分程式碼在當前的Xcode環境中已經無法執行。在WWDC16上Apple公佈了Swift3.0,從中可以看出Apple對Swift的重視,以及Swift開源半年以來的進步。儘管對於開發人員來說Swift3.0的變化會令你的程式幾乎處處報錯,但是試想一下如果Apple沒有追求極致的精神又怎麼會做出如此多的更改。今天的文章將重點介紹:What’s new in Swift 3.0?從Swift編譯器和標準庫兩個方面來說明從Swift3.0的變化。

編譯器和語法變化

函式或方法引數

  • 呼叫函式或方法時從第一個引數開始就必須指定引數名

在Swift的歷史版本中出現過在呼叫函式時不需要指定任何函式引數(或者從第二個引數開始指定引數名),在呼叫方法時則必須從第二個引數開始必須指定引數名等多種情況,而在Swift3.0中不管是函式還是方法都必須從第一個引數開始必須指定引數名(當然可以使用“_”明確指出呼叫時省略引數)。

// 從第一個引數就必須指定引數名,除非使用"_"明確指出省略引數
func sum(num1:Int,num2:Int)->Int{
    return num1 + num2
}

sum(num1: 1
, num2: 2) // old: sum(1,2)或者sum(1, num2: 2)
  • 取消var引數
//func increase(var a:Int){
//    a += 1
//}
// 上面的程式碼會報錯,可改寫成
func increase(a:Int){
    var a = a
    a += 1
}
  • inout引數修飾改放到型別前
//func increase(inout a:Int) {
//    a += 1
//}
// 上面的程式碼會報錯,可改為
func increase( a:inout Int) {
    a += 1
}

方法返回值

Swift 3.0 中方法的返回值必須有接收否則會報警告,當然其實主要目的是為了避免開發人員忘記接收返回值的情況,但是有些情況下確實不需要使用返回值可以使用”_”接收來忽略返回值。當然你也可以增加@discardableResult

宣告,告訴編譯器此方法可以不用接收返回值。

struct Caculator {
    func sum(a:Int,b:Int) -> Int {
        return a + b
    }

    @discardableResult
    func func1(a:Int,b:Int) ->Int {
        return a - b + 1
    }
}
let ca = Caculator()
ca.sum(a: 1, b: 2) // 此處會警告,因為方法有返回值但是沒有接收
let _ = ca.sum(a: 1, b: 2) // 使用"_"接收無用返回值
ca.func1(a: 1, b: 2) // 由於func1添加了@discardableResult宣告,即使不接收返回值也不會警告

可選型別

Swift3.0對於可選型別控制更加嚴謹,隱式可選型別和其他型別的運算之後獲得的是可選型別而不是隱式可選型別。

let a:Int! = 1
let b = a + 1 // 此時強制解包,b是Int型
let c = a // 注意此時c是Int? 在之前的Swift版本中c是Int!

Selector的變化

Selector的改變其實從1.0到3.0經歷了多次變化,從最早的@Selector("method:")到現在的#selector(method(param1:))可以說經歷了多次修改,好在它變得越來越好,畢竟字串操作對於語法檢查來說是很無助的。

class MyClass {
    @objc func sum(a:Int,b:Int) -> Int {
        return a + b
    }

    func func1(){
        let _ = #selector(sum(a:b:))
    }
}

// old: Swift 2.2
//class MyClass {
//    @objc func sum(a:Int,b:Int) -> Int {
//        return a + b
//    }
//    
//    func func1(){
//        let _ = #selector(sum(_:b:))
//    }
//}

協議中的可選方法

在Swift3.0之前如果要定義協議中可選方法,只需要給協議加上@objc之後方法使用optional修飾就可以了,但是Swift3.0中除了協議需要@objc修飾,可選方法也必須使用@objc來修飾。

@objc protocol MyProtocol {
    @objc optional func func1() //old: optional func func1()
    func func2()
}

取消++、–操作符

var d = 1
d++ //報錯,可以改寫成 d += 1 或者 d = d + 1

取消C風格for迴圈

//for var i = 0 ;i < 10 ; i += 1 {
//    debugPrint(i)
//}
// 上面的程式碼會報錯,可改寫成如下程式碼
for i in 0  ..< 10  {
    debugPrint(i)
}

SDK類庫變化

大家都知道Swift誕生在Objective-C已經發展的相當成熟的情況下,為了保證ObjC開發人員順利過渡到Swift,也因為Swift處於初級階段,很多類庫和方法命名都儘量和ObjC保持一致,在使用Swift開發iOS應用中處處可以看到ObjC的影子。但是作為一門Modern語言Swift還是做出了改變,從中可以看出日後Swift將徹底擺脫ObjC的影子。這其中包括重新匯入Foundation消除型別字首、方法名去重、函式和方法去C風格等等。

命名

// 1.去掉字首
let url1 = URL(string: "www.cmjstudio.com")
let isFileURL = url1?.isFileURL //old:url1.fileURL ,現在更加註重語意
let data1 = Data() //NSData

// 2.方法名使用動詞,其他名詞、介詞等作為引數或移除
var array1 = [1,2,3]
array1.append(contentsOf: [4,5,6]) // old:array1.appendContentsOf([4,5,6])
array1.remove(at: 0) // old:array1.removeAtIndex(0)

// 3.不引起歧義的情況下儘量消除重複
let color1 = UIColor.red() // old:var color1 = UIColor.redColor()

// 4.列舉成員首字母變成小寫
let label1 = UILabel()
label1.textAlignment = .center // old:label1.textAlignment = .Center

// 5.按鈕的Normal狀態去掉
let btn1 = UIButton()
btn1.setTitle("hello", for: UIControlState()) // 相當於Normal狀態

去C風格

Swift發展初期很多類庫的引入依然保持的ObjC風格,但是ObjC由於根出C語言,因此很多操作其實並不是物件和方法操作而是C語言的函式形式。到了Swift3.0之後這一現狀將發生變化,全域性函式將會變成某些型別的方法;某些常量定義將以某個列舉型別的成員來表示。

let rect1 = CGRect(x: 0, y: 0, width: 100, height: 100)
// 下面的程式碼將要報錯,3.0完全廢除這種類C的風格
//let rect1 = CGRectMake(0, 0, 100, 100)

if let context1 = UIGraphicsGetCurrentContext() {
    CGContext.fillPath(context1) // old:CGContextFillPath(context1!)
}

// GCD的改變
let queue = DispatchQueue(label: "myqueue")
queue.async {
    debugPrint("hello world!")
}
// old:
//let queue = dispatch_queue_create("myqueue", nil)
//dispatch_async(queue) {
//    debugPrint("hello world!")
//}

// 相關常量定義被移到列舉內部
NotificationCenter.defaultCenter().addObserver(self, selector: #selector(userDefaultChange()), name: UserDefaults.didChangeNotification, object: nil)
//old:NSNotificationCenter.defaultCenter().addObserver(self, selector: #selector(userDefaultChange()), name: NSUserDefaultsDidChangeNotification, object: nil)

集合API的變化

let array1 = [1,2,3]
let next = array1.index(after: 0)  // old:let start = array1.startIndex let next = start.successor()
let first = array1.first { (element) -> Bool in // 增加新的方法
    element > 1
}

let r = Range(0..<3) //old: let _ = NSRange(location: 0, length: 3)

// 下面的程式碼必須在控制器中執行,用於遍歷當前view及其父檢視
for subview in sequence(first: self.view, next: { $0?.superview }){
    debugPrint(subview)
}

新的浮點協議

Float、Double、CGFloat使用了新的協議,提供了提供 IEEE-754標準的屬性和方法。

let a = 2 * Float.pi // old: let a = 2 * M_PI
let b = 2.0 * .pi // 注意前面是浮點型,後面可以省略Float

當然Swift3.0中還有一些其他的不常用變化,如果感興趣可以訪問Swift Proposals

從Swift2.2遷移到Swift3.0

可以看出如果要更新到Swift3.0現有專案需要作出大量修改,經過使用之前開源專案TagEditor進行測試,區區十個類檔案就出現了一百多個錯誤,不過好在Xcode 8已經提供了很好用的遷移工具(Xcode:Editor - Convert - To Current Swift Syntax),經過遷移工具轉化後僅僅發現兩處錯誤需要手動修正。在使用這個工具的時候大家會看到如下介面:

Swift 2.3?沒錯Swift2.2和Swift 3.0中間還有個Swift 2.3,Apple也解釋了什麼是Swift 2.3,其實就是Swift 2.2 + New SDKs。之所以如此是因為Xcode 8目前還是beta版,使用Swift 3.0進行開發的應用還不能提交App Store,因此在Swift 2.2基礎上使用新的SDK開發還是有存在必要的。

總結

Swift的每次變化由於對之前的版本乃至上一個版本都不相容造成每次Swift的升級都顯得比較虐心,但是事實上這也是Swift的重大進步。記得之前曾有傳聞說Swift3.0的語法和API都會穩定並且向上相容,但是不久這個訊息就破滅了,WWDC上官方也再次證實這個希望可能要到4.0才能實現。但是試想一下:Apple在很短的時間內就固話API對於Swift的發展真的是好事嗎?畢竟新特性的加入、更好的語法優化才能讓Swift越來越好!總的來說,如果應用要升級到Swift3.0可能要做不同程度的修改,但是這種改動僅僅是語法和SDK的變動並不會消耗太多的工作量,更何況Apple提供了遷移工具。