Swift心得笔记之运算符

时间:2021-05-02

空值合并运算符和区间运算符

今天主要看的内容是 Swift 中的基本运算符。记录一下。

Nil Coalescing Operator

a ?? b 中的 ?? 就是是空值合并运算符,会对 a 进行判断,如果不为 nil 则解包,否则就返回 b 。

? 1 2 3 4 5 6 7 var a: String? = "a" var b: String? = "b" var c = a ?? b // "a" a = nil c = a ?? b // "b" b = nil c = a ?? b ?? "c" // "c"

使用的时候有以下两点要求:

a 必须是 optional 的
b 必须和 a 类型一致
也就是说,a 一定要有被备胎的可能,b 一定要有做备胎的资格。

其实也就是对三目运算符的简写:

复制代码 代码如下:
a != nil ? a! : b 或者 a == nil ? b : a!

当然你也可以通过自定义运算符来实现:

? 1 2 3 4 5 6 7 8 9 10 11 12 infix operator ||| {} func |||<T> (left: T?, right: T) -> T { if let l = left { return l } return right } var a:String? var b = "b" var c = a ||| b

C# 中也有个 ?? ,感兴趣的可以去了解一下。

Range Operator

区间运算符分为闭区间 (...) 和左闭右开区间 (..<) 两种,前者是算头算尾,后者是算头不算尾。

可以应用在 switch 中:

? 1 2 3 4 5 6 7 8 9 switch aNumber { case 0...5: println("This number is between 0 and 5") case 6...10: println("This number is between 6 and 10") default: println("This number is not between 0 and 10") }

区间运算符其实返回的是一个 Range<T> 对象,是一个连续无关联序列索引的集合。

话说以前左闭右开是 .. ,这样和 Ruby 的就刚好完全相反了。。。

不过有人就是想用 .. ,那么可以这样自己写一个:

? 1 2 3 4 5 6 7 8 9 infix operator .. { associativity none precedence 135} func .. (lhs: Int, rhs: Int) -> Range<Int> { return lhs..<rhs } for i in 0..10 { println("index \(i)") }

你也可以用 generate() 来遍历:

? 1 2 3 4 5 6 7 var range = 1...4 var generator = range.generate() // {startIndex 1, endIndex 5} generator.next() // 1 generator.next() // 2 generator.next() // 3 generator.next() // 4 generator.next() // nil

.generate() 返回一个 RangeGenerator<T> 的结构体,可以用来遍历 Range<T> 中的值。

以前还有个 (5...1).by(-1) 的用法,不过现在好像没用了。

区间运算符返回的是一个 ClosedInterval 或者 HalfOpenInterval 的东西,类型只要是 Comparable 就可以了。所以我们也可以把 String 放到 ... 里。

比如猫神的 Swifter Tips 中有一章的代码如下,通过 String 的 ClosedInterval 来输出字符串中的小写字母:

? 1 2 3 4 5 6 7 8 let test = "Hello" let interval = "a"..."z" for c in test { if interval.contains(String(c)) { println("\(c)") } }

SubString

Ruby 中用点点点来获取 SubString 的方法很方便:

? 1 2 3 4 5 6 2.1.3 :001 > a="abc" => "abc" 2.1.3 :002 > a[0] => "a" 2.1.3 :003 > a[0..1] => "ab"

而 Swift 中的 ClosedInterval 是没有 subscript 的。但是任性的我们就是要用 [1...3] 这种方法怎么办呢?
自己动手丰衣足食,写个 extension 吧,只需要加个 subscript 就可以了,然后下标的类型是 Range<Int> 就可以了:

? 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 extension String { subscript (r: Range<Int>) -> String { get { let startIndex = advance(self.startIndex, r.startIndex) let endIndex = advance(startIndex, r.endIndex - r.startIndex) return self[Range(start: startIndex, end: endIndex)] } } } var s = "Hello, playground" println(s[0...5]) // ==> "Hello," println(s[0..<5]) // ==> "Hello"

如果要搜索目标字符串之后再截取 substring 可以这样:

? 1 2 3 4 5 6 7 8 9 10 let name = "Joris Kluivers" let start = name.startIndex let end = find(name, " ") if (end != nil) { let firstName = name[start..<end!] } else { // no space found }

以上所述就是本文的全部内容了,希望大家能够喜欢。

声明:本页内容来源网络,仅供用户参考;我单位不保证亦不表示资料全面及准确无误,也不保证亦不表示这些资料为最新信息,如因任何原因,本网内容或者用户因倚赖本网内容造成任何损失或损害,我单位将不会负任何法律责任。如涉及版权问题,请提交至online#300.cn邮箱联系删除。

相关文章