协议定义了适合某个特定任务或功能须要的方法、属性和其他需求的一个蓝图。协议本身不提供这些需求的实现,它仅仅是描写叙述了一个任务或功能实现的蓝图。

协议与java 语言中的接口定义类似,都是描写叙述了一个实现能够干什么,而本身却不包括不论什么实现。与接口不同的是swift语言定义的协议能够被一个类、结构、或者枚举採用,来提供协议规定需求的实际实现,而java 语言的接口仅仅能被类实现。

满足一个协议需求的不论什么类型(即实现了该协议规定需求)被称为符合该协议。

协议中规定的需求能够是实例属性、实例方法、类型方法、操作符、下标方法等。

1.1 协议语法

协议的定义语法与类、结构和枚举类型类似。以protocolkeyword来指示一个协议。并在大括号里规定协议的每个需求。

protocol SomeProtocol {

// protocol definition goes here

}

一个採用某些协议的特定类型在定义中声明它採用的协议,语法例如以下:

struct SomeStructure:
FirstProtocol, AnotherProtocol {

// structure definition goes here

}

与接口类似,某个类型能够採用(符合)多个协议。类型採用的每一个协议之间用逗号来切割。类型名与协议名之间用冒号切割。

假设一个类有超类,把它的超类名放在它採用的协议名前面,并以一个逗号切割:

class SomeClass: SomeSuperclass,FirstProtocol,AnotherProtocol {

// class definition goes here

}

1.2  协议需求项定义

1) 属性需求

一个协议中的一个属性需求仅对须要属性的属性名和类型进行规定,而不规定这些属性是否是一个存储属性或者是一个计算属性。因此仅仅要一个类型提供一个和该属性需求规定的属性名和类型同样的一个实例属性或者一个类型属性,则该类型就符合该协议规定的这条属性需求。

协议的属性需求也必须规定每一个属性是仅仅读的或者是可读写的。

假设一个协议中一个属性需求规定是可读写的,那么该属性需求不能被一个常量存储属性或者一个仅仅读的计算属性满足。假设协议的一个属性需求规定是仅仅读的。那么该属性需求能被不论什么种类的属性满足。

协议中的属性需求总是作为变量属性被声明(var)。可读写属性通过在类型声明后面加入{ get set }来指示,可读属性用{ get }来指示。协议中的实例属性需求规定例如以下:

protocol SomeProtocol {

var mustBeSettable:Int {getset }

var doesNotNeedToBeSettable:Int {get }

}

在协议中总是使用classkeyword来规定一个类型属性。无论採用该协议的可能是一个结构或枚举,而在该结构或枚举中却是使用static前缀来定义该类型属性,仅仅要该类型属性的其他方面满足协议中对该属性的规定。该属性就是符合协议需求的。

protocol AnotherProtocol {

class var someTypeProperty:Int {getset }

}

2) 方法需求

协议中使用与正常的方法同样的语法来规定一条方法需求,但不同意为方法參数规定默认值。

与属性规定同样,在协议中总是使用classkeyword来规定一个类型方法需求,无论採用该协议的一个结构或枚举是使用static前缀来定义该方法,仅仅要该方法的其他方面满足该协议对该方法的规定,其方法就是符合协议需求的。

协议的一条方法需求规定例如以下所看到的:

protocol SomeProtocol {

class fund someTypeMethod()

}

3) 变异方法需求

无论採用协议的是一个类,还是一个结构或枚举,协议总是使用mutatingkeyword来标示一个方法是变异方法。一个採用该协议的类的方法不使用mutatingkeyword来标示。

例如以下所看到的。协议中规定了一条变异方法需求。

protocol Togglable {

mutating functoggle()

}

4)协议中的选项需求

你也能在协议中定义选项需求,这些需求不须要被採用该协议的类型实现(能够实现,也能够不实现)。

选项需求使用@optionalkeyword来标示。

你能够使用选项链来检查一个选项需求是否被採用该协议的类型实现。

选项属性需求、返回一个值的选项方法需求在它们被存取或调用时总是返回一个适当类型的一个选项。

选项需求仅可以在一个使用@objc 属性进行标记的协议中规定,另外使用@objc 属性进行标记的协议仅能被类所採用。而不能被结构或枚举採用。

例如以下样例展示了协议中选项需求的定义,以及怎样使用选项链来使用协议中规定的选项:

@obj cprotocol CounterDataSource {

@optional func incrementForCount(count:Int)
->Int

@optional var fixedIncrement:Int {get }

}

该协议规定了一条选项方法,一条选项属性。

@objc class Counter {

var count =0

var dataSource:CounterDataSource?

func increment()
{

if let amount =dataSource?.incrementForCount?

(count)
{

count +=amount

}else if let amount =dataSource?

.fixedIncrement?

{

count +=amount

}

}

}

使用该协议的类Counter也用@objc标记,在该类中使用选项链来读取和调用协议中规定的选项属性和选项方法。

1.3 协议的类型

协议是一种类型。因此能够使用协议作为一个函数、方法或初始化方法中的參数类型或者返回类型。也能够作为一个常量、变量或属性的类型。也能够作为一个数组、词典或其他容器中的项的类型。

因为协议是类型,因此它也可以使用is 和as操作符来检查协议的兼容性。可以使用is 操作符来检查一个实例是否採用了某个协议。使用as操作符来转换一个协议到它的继承链上的另外的协议。

与协议的选项需求规定类似,使用以上操作来检查协议的兼容性也须要协议使用@objc 属性进行标记,另外使用@objc 属性进行标记的协议仅可以被类採用,而不能被结构或枚举採用。

1.4 用扩展来採用协议

可以使用扩展来扩展某个类型(即使已经存在)。使该类型採用和符合一个新协议。

在扩展中实现新协议规定的方法、属性和下标。

扩展后的类型扩展前的已存在的实例也自己主动获得协议规定的能力。

语法例如以下:

extension ExistType:NewProtocol {

// protocol implementation goes here

}

一个已经实现某个协议全部需求的类型假设不明白声明採用该协议。那么它不自己主动採用那个协议。一个类型採用某个协议。必须被声明。

能够使用一个空的扩展来明白地声明已实现某个协议全部需求的类型採用这个协议。

extension ExistType: NewProtocol {}

1.5 协议的继承

与接口同意继承同样,协议也可以继承。

协议可以继承一个或多个其他的协议。使一个协议在它继承的协议基础上添加进一步的需求。协议继承的语法例如以下:

protocol InheritingProtocol:SomeProtocol,AnotherProtocol {

// protocol definition goes here

}

1.6  协议的复合

在使用协议作为类型使用时。你可以组合多个协议作为一个暂时类型使用,语法为:

protocol<SomeProtocol, AnotherProtocol>,在三角括号内列出要组合的多个协议,每一个协议间用逗号切割。

例如以下样例展示了使用协议组合作为一个函数參数类型的使用方法。

protocol Named {

var name:String {get }

}

protocol Aged {

var age:Int {get }

}

struct Person:Named,Aged {

var name:String

var age:Int

}

func wishHappyBirthday(celebrator:protocol<Named,Aged>)
{

println("Happy birthday\(celebrator.name) -
you're \(celebrator.age)!")

}

let birthdayPerson =Person(name:"Malcolm",age:21)

wishHappyBirthday(birthdayPerson)

协议复合未定义一个新的协议,而仅仅是暂时组合多个协议作为一个暂时协议类型使用。

版权全部,请转载时清楚注明链接和出处。谢谢。

版权声明:本文博客原创文章。博客,未经同意,不得转载。

苹果公司的新的编程语言 Swift 高级语言(十五)--协议的更多相关文章

  1. 苹果公司的新的编程语言 Swift 高级语言(十一)--初始化类的析构函数的一个实例

    一 .实例的初始化          实例的初始化是准备一个类.结构或枚举的实例以便使用的过程. 初始化包含设置一个实例的每个存储属性为一个初始值,以及运行不论什么其他新的实例可以使用之前须要的设置或 ...

  2. 苹果公司的新的编程语言 Swift 高级语言()两--基本数据类型

    一  .   常量和变量 Swift语言 对常量和变量的声明进行了明白的区分 Swift语言的常量类型比C 语言的constants类型更加强大,语义更加明白. 常量和变量的差别是常量在设置或初始化后 ...

  3. 苹果新的编程语言 Swift 语言进阶(十五)--协议

    协议定义了适合某个特定任务或功能需要的方法.属性和其它需求的一个蓝图.协议本身不提供这些需求的实现,它只是描述了一个任务或功能实现的蓝图. 协议与java 语言中的接口定义类似,都是描述了一个实现可以 ...

  4. 苹果新的编程语言 Swift 语言进阶(十六)--泛型

    泛型允许你定义一个宽松.可重用的函数或者类型,使用泛型能够避免代码的重复,也能以更清楚和抽象的方式来表达程序的意图. 泛型是Swift语言提供的强大功能之一,Swift提供的许多标准库都使用了泛型来创 ...

  5. 苹果新的编程语言 Swift 语言进阶(一)--综述

    Swift 是苹果开发和提供的供开发IOS 和OS X应用的一门新的语言.Swift语言基于C 和Objective-C语言,除了提供C 和Objective-C语言具有的所有语法功能外,为了编程方便 ...

  6. 苹果新的编程语言 Swift 语言进阶(三)--基本运算和扩展运算

    一 基本操作运算 1. 赋值操作 在Swift 中,能够使用赋值操作为一个常量或一个变量赋值,也能够使用多元组一次为多个常量或变量赋值. Swift 的赋值操作与其他语言最大的不同是赋值操作除了可以为 ...

  7. 苹果新的编程语言 Swift 语言进阶(十四)--扩展

    扩展是为一个已经存在的类.结构.枚举类型添加新功能的一种方式,包括为不能存取源代码的那些已经存在的类型添加功能. 扩展类似于Objective-C语言中的类别,与类别不同的是Swift语言的扩展没有名 ...

  8. 苹果新的编程语言 Swift 语言进阶(十一)--实例的初始化与类的析构

    一 .实例的初始化          实例的初始化是准备一个类.结构或枚举的实例以便使用的过程.初始化包括设置一个实例的每一个存储属性为一个初始值,以及执行任何其它新的实例能够使用之前需要的设置或初始 ...

  9. 苹果新的编程语言 Swift 语言进阶(十)--类的继承

    一.类的继承 类能够从其它类继承方法.属性以及其它特性,当一个类从另外的类继承时,继承的类称为子类,它继承的类称为超类.在Swift中,继承是类区别与其它类型(结构.枚举)的基础行为. 1.1 .类的 ...

随机推荐

  1. linux+win7双系统重装win7修复grub的办法(转)

    本人是debian+win7的双系统, 下面介绍下重装win7的整个过程以及遇到的一些小问题,在查阅相关博客和朋友的帮助下成功修复, 记录下以便以后有不时之需, 也希望能帮助到遇到同样问题的朋友! 首 ...

  2. UVa 10190 - Divide, But Not Quite Conquer!

    称号:给你第一个任期的等比数列和倒数公比,最后一个条目假定1这一系列的输出,否则输出Boring!. 分析:数学.递减的.所以公比的倒数一定要大于1.即m > 1. 然后在附加一个条件n  &g ...

  3. Codeforces 487C. Prefix Product Sequence 逆+结构体

    意甲冠军: 对于数字n, 他询问是否有1~n置换 这种布置能够在产品上模每个前缀n 有可能0~n-1 解析: 通过观察1肯定要在首位,n一定要在最后 除4意外的合数都没有解 其它质数构造 a[i]=i ...

  4. POJ 3280 间隔DP

    字符串,每次插入或删除字符需要一定的价格,问:我怎样才能使这个字符串转换成字符串回文,花最少. 间隔DP 当DP到区间[i,j+1]时,我们能够在i-1的位置加入一个str[j+1]字符,或者将在j+ ...

  5. 连载:面向对象的葵花宝典:思考、技巧与实践(39) - 设计原则 vs 设计模式

    它的设计原则,和设计模式,是否该用它? ============================================================================= 在& ...

  6. 关于java socket(转)

    1. 关于new Socket()中参数的理解 Server端: 调用ServerSocket serverSocket = new ServerSocket(1287,2);后Server端打开了指 ...

  7. HDU 3366 Passage (概率DP)

    Passage Problem Description Bill is a millionaire. But unfortunately he was trapped in a castle. The ...

  8. ABP应用层——应用服务(Application services)

    ABP应用层——应用服务(Application services) 点这里进入ABP系列文章总目录 基于DDD的现代ASP.NET开发框架--ABP系列之15.ABP应用层——应用服务(Applic ...

  9. Linux下的变化的主机名步骤

    Linux下变化的主机名gpdb 步骤1.执行vi /etc/sysconfig/network命令 NETWORKING=yesHOSTNAME=gpdb 第二步.执行hostname gpdb令 ...

  10. SQL Server编程系列(1):SMO介绍

    原文:SQL Server编程系列(1):SMO介绍 续篇:SQL Server编程系列(2):SMO常用对象的有关操作 最近在项目中用到了有关SQL Server管理任务方面的编程实现,有了一些自己 ...