当前位置: 首页 > news >正文

Swift 开发教程系列 - 第12章:协议与协议扩展

协议(Protocol)是 Swift 的一种重要特性,它定义了实现特定功能的方法、属性或其他要求。通过协议,可以将行为定义从具体实现中分离,使代码更具可读性和扩展性。Swift 的协议支持协议扩展,这一特性允许我们为协议定义默认实现,使得协议不仅仅是一个要求的集合,还可以具备部分功能。

12.1 什么是协议

协议定义了一组用于实现特定功能的方法或属性。任何符合该协议的类型都必须实现这些方法和属性,确保符合协议的类型拥有相似的功能和行为。

协议示例

protocol Drivable {var speed: Double { get set }func drive()
}class Car: Drivable {var speed: Double = 0.0func drive() {print("Driving at \(speed) km/h")}
}let myCar = Car()
myCar.speed = 80.0
myCar.drive()  // 输出:"Driving at 80.0 km/h"

在上例中,Drivable 协议要求任何符合该协议的类型都必须实现 speed 属性和 drive() 方法。Car 类遵循 Drivable 协议,并提供了具体的实现。

12.2 协议中的属性和方法

协议不仅可以定义方法,还可以定义属性和下标(subscript)。协议中的属性可以是只读的,也可以是可读写的。

  1. 只读属性:使用 { get } 声明只读属性,符合该协议的类型必须实现此属性。
  2. 可读写属性:使用 { get set } 声明可读写属性,符合该协议的类型必须支持读取和写入。

示例代码

protocol Identifiable {var id: String { get }
}struct User: Identifiable {var id: String
}let user = User(id: "12345")
print("User ID: \(user.id)")  // 输出:"User ID: 12345"

在上例中,Identifiable 协议定义了一个只读属性 id,符合协议的类型 User 实现了此属性。

12.3 协议的继承

Swift 的协议支持继承,可以从一个协议继承多个协议,并添加自己的要求。

示例代码

protocol Vehicle {var maxSpeed: Double { get }
}protocol Drivable: Vehicle {func drive()
}class Bicycle: Drivable {var maxSpeed: Double = 25.0func drive() {print("Cycling at a safe speed.")}
}let bike = Bicycle()
bike.drive()  // 输出:"Cycling at a safe speed."

在上例中,Drivable 协议继承自 Vehicle 协议,这意味着任何符合 Drivable 的类型必须同时满足 Vehicle 的要求。

12.4 协议组合

Swift 支持将多个协议组合在一起,定义一个新类型,该类型必须同时符合多个协议的要求。

示例代码

protocol Named {var name: String { get }
}protocol Aged {var age: Int { get }
}struct Person: Named, Aged {var name: Stringvar age: Int
}func printInfo(of person: Named & Aged) {print("\(person.name) is \(person.age) years old")
}let alice = Person(name: "Alice", age: 30)
printInfo(of: alice)  // 输出:"Alice is 30 years old"

在上例中,函数 printInfo 的参数类型 Named & Aged 表示参数必须同时符合 Named 和 Aged 协议。

12.5 协议扩展

协议扩展允许你为协议提供默认实现,使得符合该协议的类型可以直接使用这些实现,而无需自行实现。这样可以避免代码重复,并使协议的行为更为一致。

示例代码

protocol Greetable {var name: String { get }func greet()
}extension Greetable {func greet() {print("Hello, \(name)!")}
}struct Friend: Greetable {var name: String
}let friend = Friend(name: "John")
friend.greet()  // 输出:"Hello, John!"

在上例中,Greetable 协议的扩展提供了 greet() 方法的默认实现,因此符合协议的 Friend 结构体可以直接调用 greet() 方法,而无需自己实现。

12.6 协议的应用场景

  1. 接口设计:通过协议定义模块的公共接口,使代码更具模块化和可读性。
  2. 依赖注入:使用协议替代具体类型,可以实现更灵活的依赖注入。
  3. 委托模式:在委托模式中,协议用于定义委托对象的方法和属性,使不同类型的对象可以通过协议进行交互。
  4. 类型约束:在泛型编程中,使用协议作为类型约束,确保泛型类型符合特定要求。

委托模式示例

protocol DataSource {func fetchData() -> [String]
}class TableView {var dataSource: DataSource?func reloadData() {if let data = dataSource?.fetchData() {print("Data: \(data)")}}
}class DataProvider: DataSource {func fetchData() -> [String] {return ["Item 1", "Item 2", "Item 3"]}
}let tableView = TableView()
tableView.dataSource = DataProvider()
tableView.reloadData()  // 输出:"Data: ["Item 1", "Item 2", "Item 3"]"

在上例中,DataSource 协议定义了 fetchData() 方法,使 TableView 类不需要知道具体的数据来源,而是通过协议获取数据。

12.7 协议的优点

  1. 灵活性:协议可以帮助实现松耦合的设计,使代码更具模块化和灵活性。
  2. 可扩展性:协议扩展使得协议不只是一个要求的集合,还可以包含默认实现,方便扩展协议的功能。
  3. 代码复用:通过协议和协议扩展,可以避免重复代码,提高代码的复用性。
  4. 类型安全:Swift 的类型检查确保协议实现的安全性,避免了运行时错误。

通过本章的学习,你将会发现协议在 Swift 开发中扮演着至关重要的角色,使代码更加灵活、可扩展,并且容易维护。下一章将探讨 Swift 的错误处理机制(Error Handling),帮助你编写更加健壮和稳定的代码。

相关文章:

Swift 开发教程系列 - 第12章:协议与协议扩展

协议(Protocol)是 Swift 的一种重要特性,它定义了实现特定功能的方法、属性或其他要求。通过协议,可以将行为定义从具体实现中分离,使代码更具可读性和扩展性。Swift 的协议支持协议扩展,这一特性允许我们为…...

麒麟V10,arm64,离线安装docker和docker-compose

文章目录 一、下载1.1 docker1.2 docker-compose1.3 docker.service 二、安装三、验证安装成功3.1 docker3.2 docker-compose 需要在离线环境的系统了里面安装docker。目前国产化主推的是麒麟os和鲲鹏的cpu,这块的教程还比较少,记录一下。 # cat /etc/ky…...

NUXT3学习日记二(样式配置、引入组件库、区分在服务端还是在客户端渲染)

上一章已经给大家分享官网下载的nuxt3了,下面正式进入我所要说的内容吧 一、初始化样式 想必大家从我的git下载下来的nuxt3,能看到nuxt.config.ts这个文件了吧。 这里我们有两种css配置方式 1、css:[~/assets/base.scss] 这种方式不能在scss文件中定义…...

FPGA/Verilog,Quartus环境下if-else语句和case语句RT视图对比/学习记录

基本概念 RTL(Register - Transfer - Level)视图:是一种硬件描述语言的抽象层次,用于描述数字电路中寄存器之间的数据传输和操作。在这个层次上,可以看到电路的基本结构,如寄存器、组合逻辑、多路复用器等…...

Javascript高级—闭包问题

闭包问题 循环中赋值为引用的问题 for (var i 1; i < 5; i) {setTimeout(function timer() {console.log(i)}, i * 1000) }解决方法有3种 第一种&#xff0c;使用立即执行函数方式 for (var i 1; i < 5; i) {(fuction(j){setTimeout(function timer() {console.log…...

C#入门 017 字段,属性,索引器,常量

字段&#xff0c;属性&#xff0c;索引器&#xff0c;常量都表示数据 字段 什么是字段 字段(field)是一种表示与对象或类型(类与结构体)关联的变量字段是类型的成员&#xff0c;又称“成员变量&#xff0c;写在类体里面与对象关联的字段亦称“实例字段&#xff0c;表示某个对…...

磐石云语音助手拦截介绍

呼叫中心用户实际应用场景下最高会有超过30%的和语音助手&#xff1b;无声主要是进入了语音信箱;如&#xff1a;“听到滴声后留言”&#xff0c;”漏话提醒““发送请按1&#xff0c;重录请按2”以及拨打过程中客户主动拒接产生的”您拨打的用户正忙“&#xff0c;”关机“”停…...

JSP执行过程及其与Servlet执行效率的比较

JSP&#xff08;Java Server Pages&#xff09;和Servlet都是Java Web开发中常用的技术&#xff0c;它们都用于动态生成Web页面。然而&#xff0c;JSP和Servlet在执行过程和效率上存在一些差异。本文将详细探讨JSP的执行过程&#xff0c;并比较JSP与Servlet的执行效率。 一、J…...

open3d

open3d open3d用于 3D 数据处理的现代库。 简介 Open3D 是一个开源库&#xff0c;支持快速开发处理 3D 数据的软件。Open3D 前端公开了一组精心挑选的 C 和 Python 数据结构和算法。后端经过高度优化&#xff0c;并设置为并行化。Open3D 是从零开始开发的&#xff0c;具有一更…...

Vue中优雅的使用Echarts的三种方式

一、原始方法直接使用 1、安装ECharts&#xff1a; npm install echarts --save 2、创建一个Vue公共组件 EChart.vue&#xff1a; <template><div :style"{width: 100%, height: 300px}" ref"chart"></div> </template><scri…...

SpringBoot配置文件/日志

目录 一,SpringBoot配置文件 1,配置文件的格式: 2,properties 3,yml 1,properties与yml的转换 2,读取配置选哪个中的内容 3,单双引号的差异: 4,配置对象: 5,配置集合/配置map 6,yml的优缺点: 二,验证码: 学习目的: 实现样例: 接口定义: 代码总结: 三,日志: 1,概…...

微服务架构面试内容整理-SpringCloud Netflix‌与Spring Cloud Alibaba比较

Spring Cloud Netflix 和 Spring Cloud Alibaba 都是用于构建微服务架构的解决方案,但它们在设计理念、组件和使用场景上存在一些差异。以下是它们的比较: 1. 服务注册与发现 ● Spring Cloud Netflix:使用 Eureka 作为服务注册和发现的组件。Eureka 是基于 REST 的,适合服…...

JDBC魔法:连接MySQL数据库与数据操作的秘籍

文章目录 一. JDBC介绍二. 数据库驱动1.DriverManager2.Connection3.PreparedStatement4.ResultSet 三. JDBC连接MySQL1. 加载驱动2. 获得连接3. 关闭连接 四. JDBC实现数据新增五. JDBC实现数据删除 一. JDBC介绍 JDBC&#xff08;Java Database Connectivity&#xff09;是Ja…...

深入了解Scratch:引导初学者开启编程之旅

引言 在当今数字化时代&#xff0c;编程已不再是成年人的专利&#xff0c;而是一项逐渐向低年龄段普及的技能。Scratch作为一款面向儿童和青少年的图形化编程语言&#xff0c;以其简单易用且充满趣味性的特点&#xff0c;成为了许多初学者学习编程的首选工具。本文将深入探讨S…...

js复制内容到剪切板

复制内容到剪切板 最近预报一个需求需要点击按钮复制当前到行的内容到剪切板&#xff0c; 所以写了这个工具 export function copyText(text: string, prompt: string | null 已成功复制到剪切板!) {if (navigator.clipboard) {return navigator.clipboard.writeText(text).th…...

代码 RNN原理及手写复现

29、PyTorch RNN的原理及其手写复现_哔哩哔哩_bilibili 笔记连接: https://pan.baidu.com/s/1_Sm7ptEiJtTTq3vQWgOTNg?pwd2rei 提取码: 2rei import torch import torch.nn as nn bs,T2,3 # 批大小&#xff0c;输入序列长度 input_size,hidden_size 2,3 # 输入特征大小&a…...

企业官网的在线客服,如何提高效果?

企业官网的在线客服&#xff0c;如何提高效果&#xff1f; 作者&#xff1a;开源呼叫中心系统 FreeIPCC&#xff0c;github地址&#xff1a;https://github.com/lihaiya/freeipcc 提高企业官网在线客服的效果&#xff0c;是提升客户体验、增强客户满意度和忠诚度的关键。一个…...

「实战应用」如何可视化 DHTMLX Scheduler 中的资源工作量?

DHTMLX Scheduler是一个全面的 UI 组件&#xff0c;用于处理面向业务的 Web 应用程序中复杂的调度和任务管理需求。但是&#xff0c;某些场景可能需要自定义解决方案。例如&#xff0c;如果项目的资源&#xff08;即劳动力&#xff09;有限&#xff0c;则需要确保以更高的精度分…...

论文阅读《BEVFormer》

BEVFormer: Learning Bird’s-Eye-View Representation from Multi-Camera Images via Spatiotemporal Transformers 目录 摘要1 介绍2 相关工作2.1 基于Transformer的2D感知 摘要 3D视觉感知任务对于自动驾驶系统至关重要&#xff0c;包括基于多相机图像的3D检测和地图分割。…...

sql专题 之 sql的执行顺序

文章目录 sql的执行顺序sql语句的格式实际的执行顺序&#xff1a;虚拟表 vs 数据集虚拟表 结果集总结嵌套查询在sql查询中的执行顺序 前文我们了解了sql常用的语句&#xff0c;这次我们对于这些语句来个小思索 戳这里→ sql专题 之 常用命令 sql的执行顺序 SQL语句的执行顺序是…...

网络六边形受到攻击

大家读完觉得有帮助记得关注和点赞&#xff01;&#xff01;&#xff01; 抽象 现代智能交通系统 &#xff08;ITS&#xff09; 的一个关键要求是能够以安全、可靠和匿名的方式从互联车辆和移动设备收集地理参考数据。Nexagon 协议建立在 IETF 定位器/ID 分离协议 &#xff08;…...

多云管理“拦路虎”:深入解析网络互联、身份同步与成本可视化的技术复杂度​

一、引言&#xff1a;多云环境的技术复杂性本质​​ 企业采用多云策略已从技术选型升维至生存刚需。当业务系统分散部署在多个云平台时&#xff0c;​​基础设施的技术债呈现指数级积累​​。网络连接、身份认证、成本管理这三大核心挑战相互嵌套&#xff1a;跨云网络构建数据…...

如何在看板中有效管理突发紧急任务

在看板中有效管理突发紧急任务需要&#xff1a;设立专门的紧急任务通道、重新调整任务优先级、保持适度的WIP&#xff08;Work-in-Progress&#xff09;弹性、优化任务处理流程、提高团队应对突发情况的敏捷性。其中&#xff0c;设立专门的紧急任务通道尤为重要&#xff0c;这能…...

Mobile ALOHA全身模仿学习

一、题目 Mobile ALOHA&#xff1a;通过低成本全身远程操作学习双手移动操作 传统模仿学习&#xff08;Imitation Learning&#xff09;缺点&#xff1a;聚焦与桌面操作&#xff0c;缺乏通用任务所需的移动性和灵活性 本论文优点&#xff1a;&#xff08;1&#xff09;在ALOHA…...

云原生玩法三问:构建自定义开发环境

云原生玩法三问&#xff1a;构建自定义开发环境 引言 临时运维一个古董项目&#xff0c;无文档&#xff0c;无环境&#xff0c;无交接人&#xff0c;俗称三无。 运行设备的环境老&#xff0c;本地环境版本高&#xff0c;ssh不过去。正好最近对 腾讯出品的云原生 cnb 感兴趣&…...

Qemu arm操作系统开发环境

使用qemu虚拟arm硬件比较合适。 步骤如下&#xff1a; 安装qemu apt install qemu-system安装aarch64-none-elf-gcc 需要手动下载&#xff0c;下载地址&#xff1a;https://developer.arm.com/-/media/Files/downloads/gnu/13.2.rel1/binrel/arm-gnu-toolchain-13.2.rel1-x…...

【LeetCode】3309. 连接二进制表示可形成的最大数值(递归|回溯|位运算)

LeetCode 3309. 连接二进制表示可形成的最大数值&#xff08;中等&#xff09; 题目描述解题思路Java代码 题目描述 题目链接&#xff1a;LeetCode 3309. 连接二进制表示可形成的最大数值&#xff08;中等&#xff09; 给你一个长度为 3 的整数数组 nums。 现以某种顺序 连接…...

Spring Security 认证流程——补充

一、认证流程概述 Spring Security 的认证流程基于 过滤器链&#xff08;Filter Chain&#xff09;&#xff0c;核心组件包括 UsernamePasswordAuthenticationFilter、AuthenticationManager、UserDetailsService 等。整个流程可分为以下步骤&#xff1a; 用户提交登录请求拦…...

C++_哈希表

本篇文章是对C学习的哈希表部分的学习分享 相信一定会对你有所帮助~ 那咱们废话不多说&#xff0c;直接开始吧&#xff01; 一、基础概念 1. 哈希核心思想&#xff1a; 哈希函数的作用&#xff1a;通过此函数建立一个Key与存储位置之间的映射关系。理想目标&#xff1a;实现…...

macOS 终端智能代理检测

&#x1f9e0; 终端智能代理检测&#xff1a;自动判断是否需要设置代理访问 GitHub 在开发中&#xff0c;使用 GitHub 是非常常见的需求。但有时候我们会发现某些命令失败、插件无法更新&#xff0c;例如&#xff1a; fatal: unable to access https://github.com/ohmyzsh/oh…...