Nie jesteś jedynym, który nie mógł znaleźć rozwiązania.
String
nie implementuje RandomAccessIndexType
. Prawdopodobnie dlatego, że umożliwiają znaki o różnych długościach bajtów. Dlatego musimy użyć string.characters.count
( count
lub countElements
w Swift 1.x), aby uzyskać liczbę znaków. Dotyczy to również pozycji. Jest _position
to prawdopodobnie indeks do surowej tablicy bajtów i nie chcą tego ujawniać. Ma String.Index
to na celu ochronę nas przed dostępem do bajtów w środku znaków.
Oznacza to, że każdy indeks, który otrzymujesz, musi zostać utworzony z String.startIndex
lub String.endIndex
( String.Index
implementuje BidirectionalIndexType
). Wszelkie inne indeksy można tworzyć za pomocą successor
lub predecessor
metod.
Teraz, aby pomóc nam z indeksami, istnieje zestaw metod (funkcje w 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
Praca z String.Index
jest uciążliwa, ale używanie otoki do indeksowania według liczb całkowitych (patrz https://stackoverflow.com/a/25152652/669586 ) jest niebezpieczne, ponieważ ukrywa nieefektywność rzeczywistego indeksowania.
Należy pamiętać, że implementacja szybkiego indeksowania ma problem polegający na tym, że indeksy / zakresy utworzone dla jednego ciągu nie mogą być niezawodnie użyte dla innego ciągu , na przykład:
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]