(实验)Swift 面向协议

协议

swift中协议中可以声明属性,静态属性,方法,静态方案

protocol IPerson {
    //属性 协议总是用 var 关键字来声明变量属性,
    //在类型声明后加上 { set get } 来表示属性是可读可写的,可读属性则用 { get } 来表示:
    var name:String{get set}
    //static属性
    static var address:String{get set}
    //方法
    func eat()
    //static方法
    static func helloWord()
}

class Chinese: IPerson {
    var name: String = ""
    static var address: String = "china"
    func eat() {
        print("Chinese eating now......")
    }
    static func helloWord() {
        print("helloWord......")
    }
}

子类如果继承了协议而不去实现其中的内容,编译就会通不过

协议的extension实现

在Swift中 协议可以直接由协议的extension实现,继承协议的子类可以直接调用,看Demo

protocol IPerson {

}

extension IPerson
{
    func speak()
    {
        print("speaking now......")
    }
}

class Chinese: IPerson {

}

let anson = Chinese()
anson.speak()

如代码所示,我们甚至在IPerson不声明speak函数,子类就能调用

可选的协议

协议可以定义可选要求,遵循协议的类型可以选择是否实现这些要求。在协议中使用 optional 关键字作为前缀来定义可选要求。可选要求用在你需要和 Objective-C 打交道的代码中。协议和可选要求都必须带上@objc属性。标记 @objc 特性的协议只能被继承自 Objective-C 类的类或者 @objc 类遵循,其他类以及结构体和枚举均不能遵循这种协议。

@objc protocol IAnimal {
    @objc optional func sprak()
    @objc optional var name:String{get}
    @objc optional func variety() -> String
}

class Dog: IAnimal
{
    func jump() {
        print("Dog jump....")
    }
    
    var name: String{
        return "dog"
    }
    
    func variety() -> String {
        return "black"
    }
}

let dog:IAnimal = Dog()
print(dog.name ?? "")
print(dog.variety!())

协议中的可选要求可通过可选链式调用来使用,因为遵循协议的类型可能没有实现这些可选要求。
需要注意的是整个函数类型是可选的,而不是函数的返回值。

associatedtype

在Swift中,class、struct、enums都可以是用参数化类型来表达泛型的,只有在协议中需要使用associatedtype关键字来表达参数化类型

协议在Swift中有两个目的,第一个目的是用来实现多继承(swift语言被设计为单继承的),第二个目的是强制实现者必须准守自己所指定的泛型约束。关键字associatedtype是用来实现第二个目的的

泛型的实现方式

下面例子中的associatedtype实现了类似泛型的作用

protocol ICooker {
    associatedtype foodType
    func barbecue(food:foodType)    //烤*
    func saute(food:foodType)    //炒*
    
}

class Chicken {}
class Corn {}

//高级厨师可以烤肉或者炒肉
class SuperCooker: ICooker {
    typealias FoodType = Chicken
    func barbecue(food:FoodType)    //烤肉
    {
    }
    func saute(food:FoodType)       //炒肉
    {
    }
}
//一般厨师只会烤玉米或者炒玉米
class GeneralCooker: ICooker {
    typealias FoodType = Corn
    func barbecue(food:FoodType)    //烤玉米
    {
    }
    func saute(food:FoodType)       //炒玉米
    {
    }
}

判断是否符合协议

这个直接看代码

protocol ITest {
    var name:String {get set}
}

class Test: ITest {
    var name = "I am Test"
}

class Dog {
    var name = "I am dog"
}

let arr:[AnyObject] = [Dog(),Test()]

arr.forEach { (item) in
    if let item = item as? ITest {  //用强转来判断
        print("find ITest: " + item.name)
    } else {
        print("not find")
    }
}

输出

not find
find ITest: I am Test

协议的约束

限定允许集成协议的类型

protocol 后面加上约束关键字【where】,并注明该协议只能被某个类(包括子类)所遵守

比如

class Person {}

protocol IPerson where Self : Person {}

extension IPerson
{
}

class Chinese: Person,IPerson {}

//class Dog:IPerson{} 编译不通过

上面的Dog继承IPerson 编译器报错'IPerson' requires that 'Dog' inherit from 'Person'
这个限定对于协议也可以

protocol ISpeak {}
protocol IPerson where Self : ISpeak {}

extension IPerson
{
}

class Chinese: ISpeak,IPerson {}

//class Dog:IPerson{} //编译不通过

上面的 Chinese继承了ISpeak协议,符合IPerson协议的条件,所以能编译通过 Dog未继承ISpeak协议,编译失败

对具体类型的扩展

上面的例子约束了子类的类型,编译器会帮我们去判断这个约束,但是有的时候我们需要对不同的类进行不同的扩展方法
比如下面

protocol ISpeak {}

class Chinese: ISpeak {}
class Dog:ISpeak{} 

我想给Chiese和Dog不同的speak方法实现,那么可以这样

protocol ISpeak {}

class Chinese: ISpeak {}
class Dog:ISpeak{}

extension ISpeak where Self : Chinese
{
   func speak()
   {
        print("Hollo word!")
    }
}

extension ISpeak where Self : Dog
{
    func speak()
    {
        print("wang wang  wang~~~")
    }
}
Chinese().speak()
Dog().speak()

输出

Hollo word!
wang wang  wang~~~

这里利用了协议的extension的约束对不同的子类进行了扩展

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,723评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,003评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,512评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,825评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,874评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,841评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,812评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,582评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,033评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,309评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,450评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,158评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,789评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,409评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,609评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,440评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,357评论 2 352

推荐阅读更多精彩内容

  • importUIKit classViewController:UITabBarController{ enumD...
    明哥_Young阅读 3,794评论 1 10
  • 136.泛型 泛型代码让你可以写出灵活,可重用的函数和类型,它们可以使用任何类型,受你定义的需求的约束。你可以写出...
    无沣阅读 1,464评论 0 4
  • 2014年的苹果全球开发者大会(WWDC),当Craig Federighi向全世界宣布“We have new ...
    yeshenlong520阅读 2,284评论 0 9
  • 盘古开天辟地,三皇五帝传,尧舜禹汤,唯我称王,群雄逐鹿,五霸争锋,战国时期,天下风云,秦皇过后,汉推文景,唐举贞观...
    上官子玉阅读 279评论 2 1
  • 我,也就是本人! 年方二八、身高168,体重48公斤! 浓眉大眼、根正苗红一枚正儿八经的上进青年! 就是那种一眼扫...
    02419091e3f2阅读 480评论 0 0