ios - String.subscript.getter 的 NSRangeException

标签 ios swift

我在Extension中写了这个mid方法。

func mid(fromIndex: Int, toIndex: Int) -> String {

    var _fromIndex = 0

    if( fromIndex >= self.count ) {
        return ""
    }
    else if( fromIndex > 0 ){
        _fromIndex = fromIndex
    }

    var _toIndex = 0

    if( toIndex >= self.count ) {
        _toIndex = self.count - 1
    }
    else if( toIndex > 0 ){
        _toIndex = toIndex
    }

    if( _fromIndex > _toIndex ){
        return ""
    }

    return String(self[self.index(self.startIndex, offsetBy: _fromIndex)...self.index(self.startIndex, offsetBy: _toIndex)])
}

并使用此方法检索用户删除的字符串。

func textView(_ textView: UITextView, shouldChangeTextIn range: NSRange, replacementText text: String) -> Bool {
    defer{
        if text.isEmpty {
            let deletedText = textView.attributedText.string.mid(fromIndex: range.lowerBound, toIndex: range.upperBound)
        }
    }
}

但它有时会抛出这个异常。

致命异常:NSRangeException *** -[NSBigMutableString characterAtIndex:]: 索引 672 越界;字符串长度 672 专门的 String.subscript.getter

在使用 String.subscript.getter 之前,我已经检查了字符串的大小。

if( toIndex >= self.count ) {
    _toIndex = self.count - 1
}

并且文本操作应该在主线程。 为什么我仍然收到此异常? 我是否误解了 Swift 中的字符串索引?

更新1

根据@Wattholm 的回答。 我正在使用这种方式从 UITextView 获取带有 NSRange 的文本。

let contents: String = textView.text

if let _range = Range(range, in: contents) {
    let deletedText = String(contents[_range])
}
else {
    logHelper.w("Failed to get deletedText from NSRange, try UITextRange.")

    if let startPosition = textView.position(from: textView.beginningOfDocument, offset: range.location),
       let endPosition = textView.position(from: startPosition, offset: range.length),
       let textRange = textView.textRange(from: startPosition, to: endPosition), 
       let deletedText = textView.text(in: textRange) {

        // do something with deletedText
    }
    else {
        logHelper.w("Failed to get deletedText from UITextRange, try NSString.")

        let deletedText = (contents as NSString).mid(fromIndex: range.lowerBound, toIndex: range.upperBound)
    }
}

但还是落入第二个else,抛出NSRangeException。 这意味着长度和索引仍然不匹配。

更新 2

尝试使用 NSString.substring 但仍然出现此异常。

let deletedText = (contents as NSString).substring(with: range)

也许 UITextView 有问题?

最佳答案

我可能是错的,但我认为这里的问题与 String 和 NSString 之间,或者 Range 和 NSRange 之间的内在差异有关。 String 的计数并不总是与 NSString 的长度相同。

可能有一种更简单的方法来做到这一点,但我尝试的是为 NSString 创建一个扩展,就像你为 String 创建一个扩展一样,然后转换 'contents'(我假设它是一个 String;那里似乎这里缺少一些代码,因为 'contents' 没有在任何地方声明)到 NSString 以便它可以正常工作,因为 textview 为我们提供了 NSRange:

extension NSString {
    func mid(fromIndex: Int, toIndex: Int) -> NSString {

        var _fromIndex = 0

        if( fromIndex >= self.length ) {
            return ""
        }
        else if( fromIndex > 0 ){
            _fromIndex = fromIndex
        }

        var _toIndex = 0

        if( toIndex >= self.length ) {
            _toIndex = self.length - 1
        }
        else if( toIndex > 0 ){
            _toIndex = toIndex
        }

        if( _fromIndex > _toIndex ){
            return ""
        }

        return NSString(string: self.substring(with: NSRange(location: _fromIndex, length: _toIndex - _fromIndex + 1)))
    }
}

对于 TextView :

func textView(_ textView: UITextView, shouldChangeTextIn range: NSRange, replacementText text: String) -> Bool {
    defer{
        if text.isEmpty {
            let deletedText = (contents as NSString).mid(fromIndex: range.lowerBound, toIndex: range.upperBound - 1)
        }
    }
}

另一种方法是将 NSRange 转换为 Range,这可能是更好的方法,特别是如果您要求字符计数与 String 在 Swift 中计算其字符的方式完全相同。 textview函数中的range: NSRange可以这样转换成Range:

让 swiftRange = Range(range, in: rangeText)

其中 rangeText 应替换为范围所指的文本属性。

关于ios - String.subscript.getter 的 NSRangeException,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/58353002/

相关文章:

javascript - iOS Safari - 禁用滚动/弹跳但保留缩放

ios - 当我在 safari 浏览器中点击顶部的返回应用程序按钮时,如何清除 ios 应用程序中的 facebook 凭据?

ios - NSFontAttributeName 未应用于 NSAttributedString

ios - 多目标 Apple Watch

ios - 接收方在 self.performSegue(withIdentifier, sender) 中没有带有标识符的 segue

ios - Scenekit写入退出没有错误

ios - 从嵌入式导航 Controller 推送 viewController 时的奇怪行为

ios - 游戏在检查 Sprite 时偶尔会崩溃

ios - 通知中心在两个 View 之间不起作用

ios - 如何使用 swift 更改 iPhone 移动 iOS 中的下载路径