-
Notifications
You must be signed in to change notification settings - Fork 4
协议 protocol
- 自定义类型声明时, 将协议名放在类型名的冒号之后来表示该类型采纳一个特定的协议. 多个协议可以使用逗号分开列出.
- 若一个类拥有父类, 将这个父类名放在其采纳的协议名之前, 并用逗号分隔.
e.g.
protocol SomeProtocol {
}
struct SomeStructure: FirstProtocol, AnotherProtocol {
}
class SomeClass: SomeSuperClass, FirstProtocol, AnotherProtocol {
}
-
协议可以要求所有遵循该协议的类型提供特定名字和类型的实例属性或类型属性. 协议并不会具体说明属性是存储型属性还是计算型属性(它只具体要求属性有特定的名称和类型). 协议同时要求一个属性必须明确是可读的或可读和可写的.
-
若协议要求一个属性为可读和可写的, 那么该属性要求不能用常量存储属性或只读计算属性来满足. 若协议只要求属性为可读的, 那么任何种类的属性都能满足这个要求, 而且如果你的代码需要的话, 该属性也可以是可写的.
-
在协议中定义类型属性时在前面添加
static
关键字. 当类的实现使用class
或static
关键字前缀声明类型属性要求时, 这个规则仍然适用.
e.g.
protocol FullyNamed {
var fullName: String { get }
}
struct Person: FullyNamed {
// 可读可写 存储型属性
var fullName: String
}
class Starship: FullyNamed {
var name: String
init(name: String) {
self.name = name
}
// 实现协议的可读属性 计算型属性
var fullName: String {
"Starship \(name)"
}
}
let john = Person(fullName: "john")
let enterprise = Starship(name: "Enterprise")
print(john.fullName)
print(enterprise.fullName)
输出结果:
john
Starship Enterprise
e.g. 类型属性
protocol SomeProtocol {
static var someTypeProperty: Int { get set }
}
-
协议可以要求采纳的类型实现指定的实例方法和类方法. 这些方法作为协议定义的一部分, 书写方式与正常实例和类的方式完全相同, 但是不需要大括号和方法的主体. 允许拥有参数, 与正常的方法使用同样的规则. 但在协议的定义中, 方法参数不能定义默认值.
-
正如类型属性要求的那样, 当协议中定义类型方法时, 你总要在其之前添加
static
关键字. 即使在类实现时, 类型方法要求使用class
或static
作为关键字前缀, 前面的规则仍然适用.
- 若你定义了一个协议的实例方法需求, 想要异变任何采用了该协议的类型实例, 只需在协议里方法的另一当中使用
mutating
关键字. 这允许结构体和枚举类型能够采用相应协议并满足方法要求.
- 协议可以要求遵循协议的类型实现指定的初始化器. 和一般的初始化器一样, 只用于将初始化器写在协议的定义当中, 只是不用谢大括号也就是初始化器的实体.
protocol SomeProtocol {
init(someParameter: Int)
}
- 你可以通过实现指定初始化器或便捷初始化器来使遵循该协议的类满足协议的初始化器要求. 在两种情况下, 你都必须使用
required
关键字修饰初始化器的实现.
e.g.
class SomeClass: SomeProtocol {
required init(someParameter: Int) {
}
}
- 如果一个子类重写了父类指定的初始化器, 并且遵循协议实现了初始化器要求, 那么就要为这个初始化器的实现添加
required
和override
两个修饰符.
e.g.
protocol SomeProtocol {
init()
}
class SomeSuperClass {
init() {
}
}
class SomeSubClass: SomeSuperClass, SomeProtocol {
required override init() {
}
}
-
在函数、方法或者初始化器里作为形式参数类型或者返回类型
-
作为常量、变量或者属性的类型
-
作为数组、字典或者其他存储器的元素的类型
- 协议可以继承一个或者多个其他协议并且可以在它继承的基础之上添加更多要求.协议继承的语法与类继承的语法相似, 只不过可以选择列出多个继承的协议, 使用逗号分隔.
- 通过添加
AnyObject
关键字到协议的继承列表, 你就可以限制协议只能被类类型采纳(不是结构体或者枚举)
e.g.
protocol SomeClassOnlyProtocol: AnyObject, SomeInheritedProtocol {
}
-
可以使用协议组合来复合多个协议到一个要求里. 协议组合行为就和你定义的临时局部协议一样拥有构成所有协议的需求. 协议组合不定义任何新的协议类型.
-
协议组合使用
SomeProtocol&AnotherProtocol
的形式.你可以列举任意数量的协议, 用和符号连接(&
).除了协议列表, 协议组合也能包含类类型, 这允许你标明一个需要的父类.
e.g.
protocol Named {
var name: String { get }
}
protocol Aged {
var age: Int { get }
}
struct Person: Named, Aged {
var name: String
var age: Int
}
func wishHappyBirthday(to celebrator: Named & Aged) {
print("Happy birthday, \(celebrator.name) you're \(celebrator.age)!")
}
let birthdayPerson = Person(name: "Malcolm", age: 21)
wishHappyBirthday(to: birthdayPerson)
输出结果:
Happy birthday, Malcolm you're 21!
- 你可以给协议定义可选要求, 这些要求不需要强制遵循协议的类型实现. 可选要求使用
optional
修饰符作为前缀放在协议的定义中. 可选要求允许你的代码与Object-C
融合操作. 协议和可选要求必须使用@objc
标志标记. 注意@objc
协议只能被继承自Object-C
类或其他@objc
类采纳. 他们不能被结构体或者枚举采纳.
ShenYj.github.io - 简书地址 - 返回首页
-
Apple
Common
蓝牙
LBS
音视频
- AVAudioSessionchange_route
- 切换线路
- StreamingKit
- Audio Unit 基础
OC 与 Swift 混编
Object-C
- 代码混淆
- autoreleasepool
- 忽略编译器(clang)警告
- 定时器
- 锁
- RunLoop
- block
- NS_REFINED_FOR_SWIFT
- NS_CLOSED_ENUM
- NS_TYPED_ENUM、NS_STRING_ENUM
- NS_TYPED_EXTENSIBLE_ENUM、NS_EXTENSIBLE_STRING_ENUM
- 关键字nonnull和nullable
- class、objc_getClass和object_getclass方法区别
- isKindOfClass和isMemberOfClass
- 应用程序的加载
- non-lazy classes & lazy classes
- load方法
- initialize方法
- 方法的本质
- 类型编码
- self和super
- 类的内存分析
Swift
- precondition
- 权限控制
- Array常用Api
- String初始化、定义
- String常用Api
- String截取演练
- Set定义、创建
- Set访问和修改
- Dictionary操作
- Dictionary和KeyValuePairs
- Dictionary与String转换
- 常用高阶函数
- enum原始值
- enum关联值
- enum遍历
- 递归enum
- enum内存分配
- 指针
- for循环
- break跳出循环
- 变量名与关键字冲突
- 类的定义
- 类的继承和初始化
- 关键字: final
- 关键字: mutating
- 关键字: lazy
- 修饰类方法的关键字: static
- 关键字: final、dynamic、objc和_dynamicReplacement
- 关键字:@dynamicMemberLookup和@dynamicCallable
- 关键字: propertyWrapper
- 自定义运算符
- 下标: subscript
- 扩展: extension
- 协议: protocol
- 协议和扩展
- 为什么需要泛型
- 泛型函数定义
- 泛型类型
- 泛型的类型约束
- 关联类型
- 为泛型定义要求
- 泛型下标
- 多线程
- Attributes
- 错误处理
- Codable
- DispatchSourceTimer
- Swift 5.x 演练: 更多功能编辑页
- Swift 5.x 类库收集
- 单元测试笔记
- 实例对象内存结构
- 元类型、Type、Self
- frozen
- convention
- Swift(5.3.2)源码编译
- SQLite.Swift类库演练
- Swift 5.5 关键字: async/await
- Swift 5.5 新特性: Continuations
- Swift 5.5 新特性: Actor
- Swift 方法调度
- Swift Mirror
- Swift 关键字: @_silgen_name
- Swift 关键字: @_disfavoredOverload
- swiftmodule
- Swift 5.6 新特性: Type placeholders
- Swift 5.6 新特性: #unavailable
- Swift 5.6 新特性: CodingKeyRepresentable
- Swift 5.6 新特性: existential any
- Swift 5.7 新特性: if-let/guard 语法简化
- Swift 5.7 新特性: Multi-statement closure type inference
- Swift 5.8 新特性: @backDeployed
- Swift 5.9 新特性: if switch expressions
- Swift 6.0 新特性:@preconcurrency
RxSwift
macOS - AppKit
-
iOS Assembly(ARM64)
-
C++
C++ 基础
- cout、cin
- 函数重载
- 默认参数
- extern "C"
- pragma once
- inline function
- const
- Reference
- 汇编
- 类和对象
- 堆空间内存管理
- Constructor
- Destructor
- 成员变量初始化
- 声明与实现分离
- namespace
- 继承
- 访问权限
- 初始化列表
- 多态:虚函数
- 多态:虚函数实现原理
- 多态:虚析构函数
- 多态:纯虚函数
- 多态:抽象类
- 多继承
- static
- static: 单例模式
- const 成员
- 引用类型成员
- 拷贝构造函数
- 调用父类的拷贝构造函数
- 浅拷贝、深拷贝
- 对象型参数和返回值
- 匿名对象
- 隐式构造
- 编译器自动生成的构造函数
- 友元
- 内部类
- 局部类
- 运算符重载
- 模板
- 类型转换
- C++标准
- Lambda
- 异常
- 智能指针
-
Flutter
Dart
Flutter
-
Go
Go 基础
-
Ruby
Ruby 基础
-
React-Native
React-Native
-
工具篇
-
Swift Package Manager
-
自动化
-
TroubleShooting
-
扩展