Вы не единственный, кто не может найти решение.
String
не реализует RandomAccessIndexType
. Вероятно, потому что они позволяют символы с различной длиной байта. Вот почему мы должны использовать string.characters.count
( count
или countElements
в Swift 1.x), чтобы получить количество символов. Это также относится к позициям. _position
, Вероятно , индекс в сырой массив байтов , и они не хотят подвергать это. Он String.Index
предназначен для защиты нас от доступа к байтам в середине символов.
Это означает, что любой индекс, который вы получаете, должен быть создан из String.startIndex
или String.endIndex
( String.Index
реализует BidirectionalIndexType
). Любые другие индексы могут быть созданы с помощью successor
или predecessor
методов.
Теперь, чтобы помочь нам с индексами, есть набор методов (функции в Swift 1.x):
Swift 4.x
let text = "abc"
let index2 = text.index(text.startIndex, offsetBy: 2) //will call succ 2 times
let lastChar: Character = text[index2] //now we can index!
let characterIndex2 = text.index(text.startIndex, offsetBy: 2)
let lastChar2 = text[characterIndex2] //will do the same as above
let range: Range<String.Index> = text.range(of: "b")!
let index: Int = text.distance(from: text.startIndex, to: range.lowerBound)
Swift 3.0
let text = "abc"
let index2 = text.index(text.startIndex, offsetBy: 2) //will call succ 2 times
let lastChar: Character = text[index2] //now we can index!
let characterIndex2 = text.characters.index(text.characters.startIndex, offsetBy: 2)
let lastChar2 = text.characters[characterIndex2] //will do the same as above
let range: Range<String.Index> = text.range(of: "b")!
let index: Int = text.distance(from: text.startIndex, to: range.lowerBound)
Swift 2.x
let text = "abc"
let index2 = text.startIndex.advancedBy(2) //will call succ 2 times
let lastChar: Character = text[index2] //now we can index!
let lastChar2 = text.characters[index2] //will do the same as above
let range: Range<String.Index> = text.rangeOfString("b")!
let index: Int = text.startIndex.distanceTo(range.startIndex) //will call successor/predecessor several times until the indices match
Swift 1.x
let text = "abc"
let index2 = advance(text.startIndex, 2) //will call succ 2 times
let lastChar: Character = text[index2] //now we can index!
let range = text.rangeOfString("b")
let index: Int = distance(text.startIndex, range.startIndex) //will call succ/pred several times
Работать с String.Index
неудобно, но использование оболочки для индексации целыми числами (см. Https://stackoverflow.com/a/25152652/669586 ) опасно, поскольку скрывает неэффективность реальной индексации.
Обратите внимание, что реализация индексации Swift имеет проблему, заключающуюся в том, что индексы / диапазоны, созданные для одной строки, нельзя надежно использовать для другой строки , например:
Swift 2.x
let text: String = "abc"
let text2: String = "🎾🏇🏈"
let range = text.rangeOfString("b")!
//can randomly return a bad substring or throw an exception
let substring: String = text2[range]
//the correct solution
let intIndex: Int = text.startIndex.distanceTo(range.startIndex)
let startIndex2 = text2.startIndex.advancedBy(intIndex)
let range2 = startIndex2...startIndex2
let substring: String = text2[range2]
Swift 1.x
let text: String = "abc"
let text2: String = "🎾🏇🏈"
let range = text.rangeOfString("b")
//can randomly return nil or a bad substring
let substring: String = text2[range]
//the correct solution
let intIndex: Int = distance(text.startIndex, range.startIndex)
let startIndex2 = advance(text2.startIndex, intIndex)
let range2 = startIndex2...startIndex2
let substring: String = text2[range2]