Как получить полный диапазон строк для заданного диапазона символов в Swift?

У Apple есть старый пример, как получить диапазон всей строки, учитывая определенный диапазон символов:

Подсчет строк текста

Чтобы получить полный диапазон первой строки, они вызывают следующую функцию Objective-C:

[string lineRangeForRange:NSMakeRange(0, 0)]

Я пытался реализовать то же самое в Swift, но не могу заставить его работать, потому что сигнатура метода изменилась:

string.lineRange(for: NSRange(location: 0, length: 0))

выдает ошибку компилятора:

Argument type 'NSRange' (aka '_NSRange') does not conform to expected type 'RangeExpression'

RangeExpression — это какой-то странный протокол, который я не совсем понял. Однако я решил, что Range<Bound> ему соответствует, поэтому попробовал следующее:

let range = NSRange(location: 0, length: 0)
textView.string.lineRange(for: Range<Int>(range)!)

На этот раз я получаю еще одну ошибку компилятора:

Generic parameter 'R' could not be inferred

Я не смог найти общий параметр R ни в Range, ни в RangeExpression.

Что это такое и как я могу заставить это работать?


person Mischa    schedule 22.02.2019    source источник


Ответы (1)


lineRange(for:) (по существу) ожидает диапазон String.Index, а не диапазон целых чисел. Вот простой пример:

let string = "Line1\nLine2"

// Find the full range of the line containing the first "1":
if let range = string.range(of: "1") {
    let lineRange = string.lineRange(for: range)
    print(string[lineRange]) // Line1
}

Фактический параметр является общим параметром типа R : RangeExpression, R.Bound == String.Index, что означает, что вы также можете передавать частичные диапазоны, например string.startIndex... или ..<string.endIndex.

Swift-версия примера кода Objective-C

NSString *string;
unsigned numberOfLines, index, stringLength = [string length];
for (index = 0, numberOfLines = 0; index < stringLength; numberOfLines++)
    index = NSMaxRange([string lineRangeForRange:NSMakeRange(index, 0)]);

было бы

let string = "Line1\nLine2"

var index = string.startIndex
var numberOfLines = 0
while index != string.endIndex {
    let range = string.lineRange(for: index..<index)
    numberOfLines += 1
    index = range.upperBound
}

print(numberOfLines)

Здесь index..<index играет роль NSMakeRange(index, 0).

Если цель состоит в том, чтобы просто подсчитать (или перечислить) общее количество строк, тогда альтернативой является использование

string.enumerateLines(invoking: { (line, _) in
    // ...
})

вместо этого (сравните Как разделить строку новыми строками в Swift).

person Martin R    schedule 22.02.2019