下面列舉的都是一些比較偏僻的,特有的,很多人不知道的協議特性。
更多查看:https://segmentfault.com/a/1190000010065953
雖然我們不能在協議中直接提供屬性和方法的默認實現,但是我們可以通過協議擴展來達到此目的。
甚至可以在擴展中添加協議里沒有定義過的方法和屬性。
在協議中使用 static 關鍵字來聲明類型方法,如果實現該協議的類型是 class 類型,則在實現類中除了用 static 來修飾類型方法外,也可以使用 class關鍵字。區別是 static 修飾的屬性或方法不能被子類復寫,class 修飾的可以被子類復寫。
如果協議中定義的實例方法會改變遵循該協議的類型的實例,那么需要在該方法前加 mutating 關鍵字, 表示可以在該方法中修改它所屬的實例以及實例的任意屬性的值。
protocol Togglable {
mutating func toggle()
}
enum OnOffSwitch: Togglable {
case off, on
mutating func toggle() {
switch self {
case .off:
self = .on
case .on:
self = .off
}
}
}
var lightSwitch = OnOffSwitch.off
lightSwitch.toggle()
// lightSwitch 現在的值為 .On
- 協議中定義get、set屬性
protocol MyProtocol {
var prop: Int { get set }
var propertyWithImplementation: String { get }
func foo()
}
extension MyProtocol {
var propertyWithImplementation: String {
return "foo"
}
func foo() {
print(prop)
}
}
class MyClass: MyProtocol {
var prop: Int = 29
}
- 協議中還可以要求遵循協議的類型實現指定的構造器:
protocol SomeProtocol {
init(someParameter: Int)
}
// 如果類為final,則不必加required標識
class SomeClass: SomeProtocol {
required init(someParameter: Int) {
// initializer implementation goes here
}
Swift 可以采用 & 符號將多個協議進行組合。
typealias Property = Named & Aged
協議還可以與類進行組合:
class City: Location, Named {
var name: String
init(name: String, latitude: Double, longitude: Double) {
self.name = name
// Location協議實例化
super.init(latitude: latitude, longitude: longitude)
}
}
- 判斷某個對象是否是某個類型用 is。在擴展協議的時候,也可以指定一些限制條件,只有遵循協議的類型滿足這些限制條件時,才能獲得協議擴展提供的默認實現。
extension Collection where Iterator.Element: TextRepresentable {
var textualDescription: String {
let itemsAsText = self.map { $0.textualDescription }
return "[" + itemsAsText.joined(separator: ", ") + "]"
}
}
let circulars = [Circular(radius: 1), Circular(radius: 2)]
print(circulars.textualDescription)
// [The circular's radius is 1, The circular's radius is 2]
- Swift 是通過協議擴展提供默認實現來到達可選的目的。但需要注意的是,標記為 @objc 的 protocol 只能被 class 實現,不能被 struct 和 enum 類型實現,而且實現它的 class 中的方法也必須被標注為 @objc,或者整個類就是繼承自 NSObject。
@objc protocol CounterDataSource {
@objc optional func incrementForCount(count: Int) -> Int
@objc optional var fixedIncrement: Int { get }
}