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

Swift 入门之自定义类型的模式匹配(Pattern Matching)

在这里插入图片描述

概览

小伙伴们都知道 Swift 是一门简洁、类型安全、极富表现力以及“性感迷人”的编程语言。

和大多数语言一样,在 Swift 中也有一些隐藏着的、不为人知的宝藏特性。利用它们我们可以极大增加撸码的愉悦和成就感。

其中,模式匹配(Pattern Matching)便是如此!征服它,我们的 Swift 开发技能又可以大步迈上一个新的台阶。

在本篇博文中,您将学到以下内容:

  • 概览
  • 1. 什么是模式匹配?
  • 2. 模式匹配符
  • 3. 模式匹配在 switch 语句中的应用
  • 4. 威力大增:模式匹配重载
  • 总结

闲言少叙,让我们马上开始模式匹配的探索之旅吧!

Let’s Go!!!😉


本篇博文对应的视频课,欢迎小伙伴们恣意观赏:

Swift趣味开发征服模式匹配(Pattern Match)

  • 哔哩哔哩 bilibili

1. 什么是模式匹配?

模式匹配是一种让我们可以更简洁更便利的比较、检查和解析数据的方式。

不光在 Swift 语言中,模式匹配的概念在很多其它编程语言中都能大放异彩。

比如,在下面 ruby 代码中我们通过模式匹配快速的检查了正则表达式是否匹配指定的字符串:

irb(main):001:0> r = /[a-z]at/
=> /[a-z]at/irb(main):002:0> r =~ "xat"
=> 0
irb(main):003:0> r =~ "11zat"
=> 2
irb(main):004:0> r =~ "1at"
=> nil

在 Swift 中的模式匹配与此类似,不过它们特别适用于 switch 语句的语境中。

那么 switch 语句如何适配模式匹配呢?答案是:用模式匹配符

2. 模式匹配符

在 Swift 语言中,模式匹配符为 ~=。在内置一些类型上它会退化为 == 操作符的行为:

let string = "hello"
let power = 11if string ~= "hello", power ~= 11 {print("matched!")
}
/* 输出结果为:
matched!
*/

注意,Swift 中匹配符(~=)与 ruby 中(=~)是正好反过来的。

我们可以为任意自定义类型做匹配模式的适配:

struct Foo {static func ~=(pattern: Double, value: Foo) -> Bool {value.size == pattern}var size = CGFloat.zero
}let foo = Foo()
if 0.0 ~= foo {print("matched!")
}
/*
输出结果为:
matched!
*/

这里有两点需要注意:

  1. 匹配模式方法可以定义在类型外面;
  2. 若还要做“逆向”匹配,则我们需要两个匹配方法;

以下是演示代码:

struct Foo {var size = CGFloat.zero
}func ~=(pattern: Double, value: Foo) -> Bool {value.size == pattern
}// 逆向匹配方法
func ~=(value: Foo, pattern: CGFloat) -> Bool {value.size == pattern
}let foo = Foo()
if 0.0 ~= foo {print("matched!")
}// 逆向匹配
if foo ~= 0.0 {print("matched too!")
}
/*输出结果为:
matched!
matched too!
*/

我们还可以对 Swift 标准库中的类型做扩展以支持匹配模式,比如让正则表达式支持 ~= 操作符:

let regex = /[a-z]at/func ~=(pattern: String, value: Regex<Substring>) -> Bool {let result = try? pattern.firstMatch(of: value)return result != nil
}if "11xat" ~= regex {print("matched!")
}
/*输出结果为:
matched!
*/

直接利用 ~= 匹配可以为我们开发带来一定便利,不过更好的 idea 是将其嵌入到 switch 语句中以发挥最大威力。

3. 模式匹配在 switch 语句中的应用

在 Swift 中 switch 语句背后那个“默默无闻的男人”恰好就是模式匹配操作符。

struct Foo {var size = CGFloat.zero
}func ~=(pattern: Double, value: Foo) -> Bool {value.size == pattern
}

对于上面已经抱上模式匹配大腿的 Foo 类型,我们可以直接将其融入到 switch 语句中去:

let foo = Foo()switch(foo) {
case 0:print("ZERO")
case 5:print("FIVE")
default:print("others")
}
/*输出结果:
ZERO
*/

4. 威力大增:模式匹配重载

如果小伙伴们以为这就结束了,那显然大家的格局还要再打开一些。

利用重载机制,我们可以让自定义类型在 switch 语句中的模式匹配更加大放异彩。

比如,如果除了直接比较 Foo#size 属性以外,我们还想判断该属性的值是否在一个范围中,这该如何是好呢?

很简单,我们对模式匹配符进行重载:

func ~=(pattern: ClosedRange<CGFloat>, value: Foo) -> Bool {pattern.contains(value.size)
}

有了上面的定义,现在我们可以在同一个 switch 语句中用两种方式来匹配 Foo 对象的值了:

let foo = Foo(size: 11)switch(foo) {
case 0:print("ZERO")
case 5:print("FIVE")
case 0...11:print("Bingooo!!!")
default:print("others")
}/* 输出结果为:
Bingooo!!!
*/

看到这里,小伙伴们是否都已摩拳擦掌,想在 Swift 中去尝试应用模式匹配更多的“奇思妙想”呢?

勇敢去爱,不畏将来,不念过往。
放手去爱,让爱如风般自由,如光般璀璨。

所以,勇敢的去吧!The brave are invincible!棒棒哒!💯

总结

在本篇博文中,我们讨论了在 Swift 中如何优雅的适配模式匹配,并介绍了如何使用模式匹配操作符重载机制在 switch 语境中让开发“简约而简单”。

感谢观赏,再会!😎

相关文章:

  • fgetc、fputs、fgets函数使用及实例——Linux编程——day2
  • Linux权限【超详细】
  • React实例之完善布局菜单(二)
  • 轻松使用python将PDF转换为图片(成功)
  • 探秘Java 8 Lambda:简洁高效的集合操作
  • STM32--HAL库定时器学习记录(易懂)--持续学习
  • 【安装指南】nodejs下载、安装与配置详细教程
  • 3D人体运动重建
  • Python循环语句——while循环的基础应用
  • 【已解决】Oracle 12541 TNS 无监听程序
  • NAS系统折腾记 – Emby搭建家庭多媒体服务器
  • SOME/IP SD 协议介绍(五)使用SOME/IP-SD宣布非SOME/IP协议的协议。
  • 深入Spring MVC的工作流程
  • 【OpenCV人脸检测】写了个智能锁屏小工具!人离开电脑自动锁屏
  • ElementUI组件:Link 文字链接
  • [LeetCode] Wiggle Sort
  • 【译】理解JavaScript:new 关键字
  • 2017届校招提前批面试回顾
  • JavaScript标准库系列——Math对象和Date对象(二)
  • Mac 鼠须管 Rime 输入法 安装五笔输入法 教程
  • Making An Indicator With Pure CSS
  • nodejs实现webservice问题总结
  • Quartz初级教程
  • react-native 安卓真机环境搭建
  • React中的“虫洞”——Context
  • RxJS: 简单入门
  • SpiderData 2019年2月25日 DApp数据排行榜
  • tweak 支持第三方库
  • Vue.js源码(2):初探List Rendering
  • Vue全家桶实现一个Web App
  • 表单中readonly的input等标签,禁止光标进入(focus)的几种方式
  • - 概述 - 《设计模式(极简c++版)》
  • 源码安装memcached和php memcache扩展
  • Java数据解析之JSON
  • ​总结MySQL 的一些知识点:MySQL 选择数据库​
  • #、%和$符号在OGNL表达式中经常出现
  • (+4)2.2UML建模图
  • (14)Hive调优——合并小文件
  • (android 地图实战开发)3 在地图上显示当前位置和自定义银行位置
  • (JSP)EL——优化登录界面,获取对象,获取数据
  • (八)五种元启发算法(DBO、LO、SWO、COA、LSO、KOA、GRO)求解无人机路径规划MATLAB
  • (附源码)ssm高校社团管理系统 毕业设计 234162
  • (论文阅读31/100)Stacked hourglass networks for human pose estimation
  • (三十五)大数据实战——Superset可视化平台搭建
  • (转载)CentOS查看系统信息|CentOS查看命令
  • .htaccess配置重写url引擎
  • .NET : 在VS2008中计算代码度量值
  • .NET Core 通过 Ef Core 操作 Mysql
  • .Net 路由处理厉害了
  • .net 怎么循环得到数组里的值_关于js数组
  • .NET下ASPX编程的几个小问题
  • [04]Web前端进阶—JS伪数组
  • [Android]通过PhoneLookup读取所有电话号码
  • [BZOJ 4598][Sdoi2016]模式字符串
  • [C/C++]数据结构 栈和队列()