Swift中Range的使用

相比OC中的NSRange,在Swift中使用Range是一个比较麻烦的事情,犹记得第一个使用,感觉写法很复杂,这里简单介绍下它的用法。

Close Ragne: a...b

这种操作创建了一个包括a和b的区间,有两种不同的闭区间,CloseRange和CountableClosedRange

  • CloseRange

在Swift中所有Ranges中的元素都是可比的,遵循Comparable协议,这让我们可以获取集合中任意区间的元素

let myRange: ClosedRange = 1...3

let myArray = ["a", "b", "c", "d", "e"]
myArray[myRange] // ["b", "c", "d"]
  • CountableClosedRange

它同CloseRange的区别就是它可以遍历,遵循了Sequence协议

let myRange: CountableClosedRange = 1...3

let myArray = ["a", "b", "c", "d", "e"]
myArray[myRange] // ["b", "c", "d"]

for index in myRange {
    print(myArray[index])
}

Half-Open Ranges: a..<b

这是个左闭右开的区间,包括a但是不包括b。同样也有两种不同的类型:Range和CountableRange

  • Range

使用同ClosedRange类似

let myRange: Range = 1..<3

let myArray = ["a", "b", "c", "d", "e"]
myArray[myRange] // ["b", "c"]
  • CountableRange
let myRange: CountableRange = 1..<3

let myArray = ["a", "b", "c", "d", "e"]
myArray[myRange] // ["b", "c"]

for index in myRange {
    print(myArray[index])
}

NSRange

在swift中同样也是需要使用NSRange,因为通过不同的含义来创建的。

let myNSRange = NSRange(location: 3, length: 2)

let myRange: Range = 3..<5

它是通过起始位置和长度来确定区间范围,不同于Range通过起始和结束的位置。

Ranges with Strings

//Range
var myString = "abcde"
let start = myString.index(myString.startIndex, offsetBy: 1)
let end = myString.index(myString.startIndex, offsetBy: 4)
let myRange = start..<end
myString.substring(with: myRange) // "bcd"

//NSRange
let myNSRange = NSRange(location: 1, length: 3)

let myNSString: NSString = "abcde"
myNSString.substring(with: myNSRange) // "bcd"

可以看到NSRange比Range要简洁多了,既然这样,为什么苹果还要想出个Range类型呢?

//Range
var myString = "a😀cde"
let start2 = myString.index(myString.startIndex, offsetBy: 1)
let end2 = myString.index(myString.startIndex, offsetBy: 4)
let myRange2 = start2..<end2
myString.substring(with: myRange2) // "😀cd"

//NSRange
let myNSString2: NSString = "a😀cde"
myNSString2.substring(with: myNSRange) // "😀c"    Where is the "d"!?

因为emoji笑脸占用了两个UTF-16单元去存储,所以和我们预期的就有所不同了。

Extension

可是要使用String时,Range写法好复杂,该怎么办?这里我们可以拓展String对象。

extension String {
    subscript(r: ClosedRange<Int>) -> String {
        let start = index(startIndex, offsetBy: r.lowerBound)
        let end = index(startIndex, offsetBy: r.upperBound)
        return self[start...end]
    }
    
    subscript(r: Range<Int>) -> String {
        get {
            let start = index(startIndex, offsetBy: r.lowerBound)
            let end = index(startIndex, offsetBy: r.upperBound)
            return self[start..<end]
        }
    }
}

//usage
"abcde"[1...3] = "bcd"
"abcde"[1..<3] = "bc"

posted @ 2017-06-19 08:43  Lawerence  阅读(5866)  评论(0编辑  收藏  举报