协议定义了一个方法的蓝图,属性和其他适合特定任务或功能的要求。协议实际上并不提供一个这些要求的实现,它只是描述了一个实现会是什么样子。协议可以通过一个类,结构或枚举提供这些要求的具体实现。满足要求的任何类型的协议都是符合协议。

协议可以要求符合类型有特定的实例属性,实例方法,类型丰富,操作符和下标。

1. 协议的语法

协议名放在类型名之后,用冒号分割,当作定义的一部分。可以列出多个协议,由逗号分隔。
如果一个类有父类,在任何协议之前列出父类名,后跟一个逗号。

protocol FirstProtocol {
    // protocol definition goes here
}
protocol AnotherProtocol {
    // protocol definition goes here
}

class SomeSuperclass{
}

class SomeClass: SomeSuperclass, FirstProtocol, AnotherProtocol {
    // protocol definition goes here
}

2. 属性要求

属性要求总是声明为变量属性,用var关键字做前缀。可获取和可设置属性是通过在他们类型声明后编写{ get set }方法,并且可获取属性是通过编写{ get }方法。
协议例子如下:

protocol ExampleProtocol {
    var simpleDescription: String { get }
    //在协议中定义方法时需要注意,类中的方法因为本身就可以改变类不需要mutating关键字修饰,但是结构struct和枚举enum使用协议时,协议和struct,enum都需要在定义和使用方法前用mutating关键字修饰
    mutating func adjust()
}
2.1 结构struct使用协议
//结构体使用协议
struct SimpleStructure: ExampleProtocol {
    var simpleDescription: String = "A simple "
    mutating func adjust() {
        simpleDescription += "struct."
    }
}
var b = SimpleStructure()
b.adjust()
let bDescription = b.simpleDescription
print(bDescription) //"A simple struct.\n"

//结构体重的方法如果需要改变结构体本身 使用的方法需要使用mutating进行修饰,如果说我们在上面的adjust中不对结构体做任何修改的话,比如说单纯的打印一段文字,不加mutating也是可以的。
2.2 类class使用协议
//类使用协议
class SimpleClass: ExampleProtocol {
    //如果在编译时不满足协议要求,Swift会报错
    var simpleDescription: String = "A simple "
    var anotherProperty: Int = 69105
    func adjust() {
        simpleDescription += "class."
    }
}
var a = SimpleClass()
a.adjust()
let aDescription = a.simpleDescription
print(aDescription) //"A simple class.\n"

//类中的方法因为本身就可以改变类所以adjust方法不需要mutating关键字修饰
2.3 枚举enum使用协议
//枚举使用协议
enum SimpleEnum:ExampleProtocol {
    case out
    mutating func adjust() {
        print(self.simpleDescription + "enum.")
    }

    var simpleDescription: String{
        get {
            switch self {
            case .out:
                return "A simple "
            }
        }
    }
}
var c = SimpleEnum.out
c.adjust()
//输出:A simple enum.

3. 方法要求

协议可以要求指定实例方法和类型方法被一致的类型实现。这些方法被写为协议定义的一部分,跟普通实例和类型方法完全一样,但是没有大括号或方法体。可变参数是允许的,普通方法也遵循同样的规则。

注意:协议为普通方法使用相同的语法,但不允许给方法参数指定默认值。

3.1 方法协议

方法协议: 定义时没有花括号执行体. 实现仅要求名称相同.

//协议内定义方法使用static进行定义,但是在实现时还是按照上面的规则:在struct或enum中仍然使用static。在class里使用class关键字时表示可override的类型方法实现,而在class里使用static关键字时表示不可override的类型方法实现
protocol MyProtocol {
    static func foo() -> String
}
struct MyStruct: MyProtocol {
    static func foo() -> String {
        return "MyStruct"
    }
}
enum MyEnum: MyProtocol {
    static func foo() -> String {
        return "MyEnum"
    }
}
class MyClass: MyProtocol {
    class func foo() -> String {
        return "MyClass,可override的类型方法实现"
    }
}
class MyClass1: MyProtocol {
    static func foo() -> String {
        return "MyClass,不可override的类型方法实现"
    }
}
3.2 构造方法协议

构造方法协议: 可以要求遵从者实现指定的构造方法.

实现时用 required init(编译器会自动提示添加)

//必须实现,实现时用 required init(编译器会自动提示添加)
protocol 带参数的构造方法协议 {
    init(某参数: Int)
}

class 某类: 带参数的构造方法协议 {
    required init(某参数: Int) { 
    }
}
protocol 构造方法协议 {
    init()
}

class 某类1: 构造方法协议 {
    required init() {
    }
}

如果子类与父类同时遵从某构造方法协议, 则子类构造方法须加override required

protocol 构造方法协议 {
    init()
}

class 父类 {
    init() {
    }
}

class 子类: 父类, 构造方法协议 {
    override required init() {
    }
}

4. 协议扩展

协议扩展: 即使无源码权限下,给已有的类添加协议.

//1.既存实例会自动遵从添加了的协议.
protocol ProtocolForInt {
    var description:String {get}
    mutating func addOne()
}
extension Int:ProtocolForInt {
    mutating func addOne() {
        self += 1
    }

    var description: String {
        return "This number is \(self)"
    }
}
var abc:Int = 4
abc.addOne()    //5
print(abc.description)  //"This number is 5\n"

//2.如果一个类型预遵从了协议, 可以直接扩展协议
struct MyText {
    var text: String
    //注意:即使满足了协议要求,类型也不会自动转变,必须为它做出明显的协议声明。
    var description: String {
        return  "我的自定义显示:" + text
    }
}

//CustomStringConvertible协议内含有必须实现的var description: String参数的可获取方法
extension MyText: CustomStringConvertible {}//协议扩展

let text1 = MyText(text: "3天学会Swift 3")

print(text1)
/*
 协议扩展时text1输出为"我的自定义显示:3天学会Swift 3\n"
 屏蔽上面的协议扩展则text1输出为"MyText(text: "3天学会Swift 3")\n"
*/

5. 协议的继承

协议能够继承一到多个协议。语法与类继承基本相同。

protocol InheritingProtocol: SomeProtocol, AnotherProtocol {
}

6. 协议实现

6.1 协议的默认实现
//自定义一个可打印和可预览协议
//CustomStringConvertible协议需满足var description: String要求
//CustomPlaygroundQuickLookable协议需满足var customPlaygroundQuickLook: PlaygroundQuickLook要求
protocol MyPrintable: CustomStringConvertible, CustomPlaygroundQuickLookable {

}

//提供默认实现: 可以给协议扩展提供一个默认的实现, 任何遵从此协议的类型都会获得.
extension MyPrintable {
    var description: String {
        return "控制台: 默认描述"
    }

    var customPlaygroundQuickLook: PlaygroundQuickLook {
        return PlaygroundQuickLook.text("playgroud预览: 默认值")
//        return PlaygroundQuickLook.color(UIColor.blue)
    }
}

//1.采用默认实现
struct MyText {
    var text: String
}
//MyText采用默认实现
extension MyText: MyPrintable {

}

let text1 = MyText(text: "abcd")    //"控制台: 默认描述\n"
print(text1)    //"控制台: 默认描述\n"

//2.不采用默认实现
struct MyText1 {
    var text: String
}
//MyText1不采用默认实现
extension MyText1: MyPrintable {
    var description: String {
        return "print时的预览:" + self.text
    }

    var customPlaygroundQuickLook: PlaygroundQuickLook {
        return PlaygroundQuickLook.text("我的Text快速预览:" + self.text)
    }
}

let text2 = MyText1(text: "efg")    //"我的Text快速预览:efg"
print(text2)    //"print时的预览:efg\n"
6.2 协议中的必需实现和可选实现方法
//协议扩展
//swift 中可选协议方法的实现
protocol SwiftDiagonProtocol{
    //必须
    func requiredMethod()

    //可选
    func optionalMethod()

}
//方法1:空实现(方法2:将可选实现和必需实现分为2个协议)
extension SwiftDiagonProtocol{
    func optionalMethod(){
        //空实现也可以
    }
}

class SwiftClass: SwiftDiagonProtocol{
    func requiredMethod() {
        //Someting
    }

    //如果需要实现可选方法则实现,这里的实现会覆盖extension中的默认实现
    func optionalMethod(){
        //Something
    }
}

7. 类专用协议

可以在协议的继承列表中,通过添加关键字『class』,限制协议只能适配到类(class)型,结构和枚举不能遵循该协议。

protocol 一个协议: class, CustomPlaygroundQuickLookable {

}

class  myText {
    var text = "22"
}

extension myText: 一个协议{
    var customPlaygroundQuickLook: PlaygroundQuickLook {
        return PlaygroundQuickLook.text(self.text)
    }
}

let text = myText() //22
text.text   //22
text.customPlaygroundQuickLook  //text("22")
print(text) //输出"myText\n",text本身仍然是myText类

8. 协议组合

协议组合: 多个协议临时组合在一起的类型. 形式: 协议1 & 协议2 & …>

protocol 年龄协议 {
    var 年龄 : Int { get }
}

protocol 姓名协议 {
    var 姓名: String { get }
}

struct  学生: 年龄协议, 姓名协议, CustomPlaygroundQuickLookable {
    var 年龄: Int
    var 姓名: String

    var customPlaygroundQuickLook: PlaygroundQuickLook {
        return PlaygroundQuickLook.text(self.姓名 + ":\(self.年龄)岁")
    }
}

//无论传递的寿星是何类型,只要遵从年龄和姓名协议即可,参数大大自由化.
func 生日祝愿(寿星: 年龄协议 & 姓名协议)  {
    print("祝",寿星.姓名,寿星.年龄,"岁生日快乐!")
}

let 学生1 = 学生(年龄: 20, 姓名: "洪荒少女")    //"洪荒少女:20岁"

生日祝愿(寿星: 学生1)   //输出:祝 洪荒少女 20 岁生日快乐!

9. 协议的检查和转换

9.1 协议的检查 is
struct Person: OneProtocol {
}
let p1 = Person()
if (p1 is OneProtocol){ //可以理解为:p1 是一个遵守了OneProtocol协议类型的实例
    print("yes")
}
//输出:yes
9.2 协议的转换 as
protocol Coder {
    var name :String {get set}
    var updating: String { get }
}

struct AndroidCoder: Coder {
    var name: String

    var updating: String {
        return "安卓程序员学iOS开发"
    }
}

struct DotNotCoder: Coder {
    var name: String
    var updating: String {
        return ".NET程序员学Swift 3"
    }
}

struct NewBie {
    var name: String
}

let a = AndroidCoder(name: "小米")
let d = DotNotCoder(name: "小猫")
let x = NewBie(name: "小狗")

let coders:[Any] = [a,d,x]

for coder in coders {
    //as操作符用来把某个实例转型为另外的类型,由于实例转型可能失败,因此Swift为as操作符提供了两种形式:选项形式as?和强制形式as!
    //as? 如果转换不成功的时候便会返回一个 nil 对象,如果能确保100%会成功的转换则可使用 as!,否则使用 as?
    //as!由于是强制类型转换,如果转换失败会报 runtime 运行错误。
    if let coder1 = coder as? Coder {
        print(coder1.updating)
    } else {
        print("你不是程序员!你会很辛苦的!")
    }

    if let xiaobo = coder as? NewBie {
        print("你是"+xiaobo.name)
    }
}
/*输出:
安卓程序员学iOS开发
.NET程序员学Swift 3
你不是程序员!你会很辛苦的!
你是小狗
*/

10. 类型约束

func findIndex<T: Equatable>(array: [T], valueToFind: T) -> Int? {
    var index: Int?

    for (i, value) in array.enumerated() {
        if (value == valueToFind) {
            index = i
        }
    }

    return index
}

var testArray = [1, 2, 3, 4]

findIndex(array: testArray, valueToFind: 3) //2

//swift标准库定义了一个Equatable协议,该协议要求任何遵循该协议的类型实现(==)和(!=)对任何两个该类型进行比较。所有的swift标准类型自动支持Equatable协议。

11. 其他

类中在协议中没有定义的属性,我们是无法访问的

protocol ProtocolForInt {
    var description:String {get}
    mutating func addOne()
}
class SimpleClassForInt:ProtocolForInt {
    var number = 0
    var description: String = "这是一个描述"
    func addOne() {
        number += 1
    }
}
var simpleClassForInt:ProtocolForInt = SimpleClassForInt()
print(simpleClassForInt.description)    //"这是一个描述\n"
simpleClassForInt.addOne()
simpleClassForInt   //number:1,description:"这是一个描述"

//print(simpleClassForInt.number) //错误,类中在协议中没有定义的属性,无法访问

我们可以看到只有协议中定义的属性和方法是我们可以访问的,类中在协议中没有定义的属性,我们是无法访问的。
尽管simpleClassForInt的运行时(Runtime)类型是SimpleClassForInt,但是编译器还是会把它视为ProtocolForInt类型,所以在协议之外的属性和方法在simpleClassForInt中是无法访问的。