- 蘋果公司在 3.29 正式釋出了正式版的
Xcode 9.3
和Swift 4.1
, 讓我們看看Swift 4.1
帶來了哪些新功能和新亮點 - 測試需要Xcode9.3, 請確保你的Xcode是最新的9.3版本
Swift 4.1
與Swift 4.0
是原始碼相容的,所以如果你已經使用Xcode
中的Swift Migrator
將你的專案遷移到Swift 4.0
,那麼新特性不會破壞你的程式碼- 下面在
Xcode9.3
下新建一個Playground
工程, 測試我們的程式碼
條件一致性(Conditional Conformance)
- 條件一致性使型別引數滿足特定條件的泛型型別的協議一致性 [ SE-0143 ]
- 在Swift 4中,如果陣列、字典或者可選型別的元素型別遵循
Equatable
,你是可以在陣列之間、字典之間和可選型別之間進行比較的, 如下示例:
// Int型別的陣列
let arr1 = [1, 2, 3]
let arr2 = [1, 2, 3]
print(arr1 == arr2)
// 比較value值為Int的字典
let dic1 = ["age": 19, "score": 60]
let dic2 = ["age": 19, "score": 60]
print(dic1 == dic2)
// 比較Int?
let age1 = dic1["age"]
let age2 = dic2["age"]
print(age1 == age2)
/// 以上輸出結果都是: true
複製程式碼
這裡如果我們把Int
都換成Int?
型別, 在Swift4.0
中是不能編譯通過的, 如下:
// Int型別的陣列
let arr1: [Int?] = [1, 2, 3]
let arr2: [Int?] = [1, 2, 3]
print(arr1 == arr2)
// 比較value值為Int的字典
let dic1: [String: Int?] = ["age": 19, "score": 60]
let dic2: [String: Int?] = ["age": 19, "score": 60]
print(dic1 == dic2)
// 比較Int?
let age1 = dic1["age"]
let age2 = dic2["age"]
print(age1 == age2)
複製程式碼
- 在這些例項中, 我們用
==
測試相等性, 在Swift4.0中,Int
型別遵循Equatable
協議, 可以比較, 但是Int?
型別卻沒有遵循Equatable
協議 - 但是在Swift4.1中, 完美的解決了這個問題, 上述程式碼可比那已通過, 且都輸出:
true
- 在
Swift 4.0
中[Set<Int>]
之間可以直接對比,但是[[Int]]
不能。現在Swift 4.1
中,[[Int]]
也能直接對比 - 總的來說,
Swift 4.1
的Array
、Dictionary
和Optional
,只要他們的元素都遵循了Equatable
和Hashable
,那麼他們也遵循Equatable
和Hashable
合成 Equatable
和 Hashable
- 如果物件相等,則這兩個物件的
hash
值一定相等 - 如果兩個物件
hash
值相等,這兩個物件不一定相等。 Swift
中Hashable
一定是Equatable
,因為前者繼承了後者。
在Swift 4
中,若遵循Equatable
協議的時候,我們必須實現Equatable
協議的==
方法,Equatable
協議如下:
public protocol Equatable {
/// Returns a Boolean value indicating whether two values are equal.
///
/// Equality is the inverse of inequality. For any values `a` and `b`,
/// `a == b` implies that `a != b` is `false`.
///
/// - Parameters:
/// - lhs: A value to compare.
/// - rhs: Another value to compare.
public static func == (lhs: Self, rhs: Self) -> Bool
}
複製程式碼
在Swift4.0中, 必須實現Equatable
協議的方法
struct Name: Equatable {
var name1 = "name1"
var name2 = "name2"
static func == (lhs: Name, rhs: Name) -> Bool {
return lhs.name1 == rhs.name1 &&
lhs.name2 == rhs.name2
}
}
複製程式碼
在Swift 4.1
,只需要加上Equatable
即可, 不需要實現任何協議方法
struct Name: Equatable {
var name1 = "name1"
var name2 = "name2"
}
複製程式碼
JSON
編碼時支援Camel Case
和Snake Case
之間的轉換
Swift 4.0
引入了Codable
,但是有個麻煩的問題:如果JSON
資料的key
命名格式是snake_case
的話,我們必須建立自己的CodingKeys
來告訴蘋果怎麼轉換。在Swift 4.0
中- 但是在
Swift 4.1
中,蘋果給JSONDecoder
引入了一個屬性keyDecodingStrategy
;對應的JSONEncoder
引入了一個屬性keyEncodingStrategy
。這樣我們就不需要設定定義CodingKeys
了。只需要在decoding
的時候把keyDecodingStrategy
設定為.convertFromSnakeCase
;在encoding
的時候把keyEncodingStrategy
設定為.convertToSnakeCase
- 下面是分別針對陣列/字典/集合的解析形式
struct Student: Codable, Hashable {
let firstName: String
let averageGrade: Int
}
let cosmin = Student(firstName: "Cosmin", averageGrade: 10)
let george = Student(firstName: "George", averageGrade: 9)
let encoder = JSONEncoder()
// Encode an Array of students
let students = [cosmin, george]
do {
try encoder.encode(students)
} catch {
print("Failed encoding students array: (error)")
}
// Encode a Dictionary with student values
let studentsDictionary = ["Cosmin": cosmin, "George": george]
do {
try encoder.encode(studentsDictionary)
} catch {
print("Failed encoding students dictionary: (error)")
}
// Encode a Set of students
let studentsSet: Set = [cosmin, george]
do {
try encoder.encode(studentsSet)
} catch {
print("Failed encoding students set: (error)")
}
// Encode an Optional Student
let optionalStudent: Student? = cosmin
do {
try encoder.encode(optionalStudent)
} catch {
print("Failed encoding optional student: (error)")
}
複製程式碼
Hashable Index Types
(雜湊化索引)
擴充套件 Key-path
表示式在標準庫中的使用範圍。讓標準庫中所有的索引型別都符合 Hashable
協議,這樣,[Int]
、String
和所有其它標準集合使用 key-path
下標時,表現都是一樣的
let swiftString2 = "one two three"
let charact1 = String.[swiftString2.startIndex]
print(swiftString2[keyPath: charact1])
let arr = [1, 2, 3, 4]
let value2 = [Int].[1]
print(arr[keyPath: value2])
//輸出結果:
o
2
複製程式碼
compactMap
的用法
在Swift 4.0
中,我們經常使用flatMap
來過濾nil
,也可以進行降維操作, 詳情可參考Swift函數語言程式設計之高階用法
let arr = [1, 2, nil, 3, 4, nil]
let arr1 = arr.flatMap({ $0 })
print(arr1)
//這樣使用會有類似的警告
`flatMap` is deprecated: Please use compactMap(_:) for the case where closure returns an optional value
Use `compactMap(_:)` instead
//Swift4.1中的用法
let arr = [1, 2, nil, 3, 4, nil]
let arr2 = arr.compactMap({ $0 })
print(arr2)
複製程式碼
主要是因為在
Swift4.0
中flatMap
有很多過載, 可能會引起歧義, 所以在Swift4.1
中把flatMap
重新命名為compactMap
除了協議中的 weak 和 unowned。
- 當你在
Tune
協議中定義了兩個屬性key
和pitch
,pitch
可能為nil
, 所以你在協議中可以用weak
修飾 - 但是如果在協議本身中定義的話,
weak
和unowned
都沒有實際意義, 所以在Swift4.1
中就已經去掉了這些關鍵字, 並且在協議中使用這些關鍵字將會爆出警告
class Key {}
class Pitch {}
// Swift 4
protocol Tune {
unowned var key: Key { get set }
weak var pitch: Pitch? { get set }
}
// Swift 4.1
protocol Tune {
var key: Key { get set }
var pitch: Pitch? { get set }
}
複製程式碼
UnsafeMutableBufferPointer
的改變
//Swift4.0
let buffer = UnsafeMutableBufferPointer<Int>(start: UnsafeMutablePointer<Int>.allocate(capacity: 10),
count: 10)
let mutableBuffer = UnsafeMutableBufferPointer(start: UnsafeMutablePointer(mutating: buffer.baseAddress),
count: buffer.count)
//Swift4.1
let buffer = UnsafeMutableBufferPointer<Int>.allocate(capacity: 10)
let mutableBuffer = UnsafeMutableBufferPointer(mutating: UnsafeBufferPointer(buffer))
複製程式碼
相對
Swift4.0
的改變,Swift4.1
這點改變鍵值太微不足道了, 傳說中Swift5
的API
會趨於穩定, 但是估計改變可能也是非常大的, 坐等Swift5
釋出…