首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >NSRangeException by String.subscript.getter

NSRangeException by String.subscript.getter
EN

Stack Overflow用户
提问于 2019-10-12 10:02:28
回答 2查看 317关注 0票数 0

我用mid编写了这个Extension方法。

代码语言:javascript
复制
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)])
}

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

代码语言:javascript
复制
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)
        }
    }
}

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

Fatal Exception: NSRangeException *** -[NSBigMutableString characterAtIndex:]: Index 672 out of bounds; string length 672 specialized String.subscript.getter

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

代码语言:javascript
复制
if( toIndex >= self.count ) {
    _toIndex = self.count - 1
}

文本操作应该在主线程上。为什么我还能得到这个例外?我是否误解了Swift中的字符串索引?

UPDATE1

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

代码语言:javascript
复制
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。这意味着长度和索引仍然不匹配。

UPDATE2

尝试使用NSString.substring,但仍然继续获得此异常。

代码语言:javascript
复制
let deletedText = (contents as NSString).substring(with: range)

也许UITextView中有一个bug?

EN

回答 2

Stack Overflow用户

发布于 2019-10-12 13:17:23

我可能是错的,但我认为这里的问题涉及字符串和NSString之间的内在区别,或者Range与NSRange之间的内在区别。字符串的计数并不总是与NSString的长度相同。

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

代码语言:javascript
复制
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)))
    }
}

对于文本视图:

代码语言:javascript
复制
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转换为一个范围,这可能是更好的方法,特别是当您要求字符计数与字符串在Swift中的字符数完全相同时。textview函数中的range: NSRange可以转换为如下所示的范围:

设swiftRange = Range(range,in: rangeText)

其中,应将rangeText替换为范围所引用的text属性。

票数 2
EN

Stack Overflow用户

发布于 2019-10-26 09:59:54

这是一个更快捷的版本,它利用了从NSRangeRange<String.Index>的可靠转换,反之亦然。

代码语言:javascript
复制
extension String {

    func mid(fromIndex: Int, toIndex: Int) -> String {
        guard toIndex >= fromIndex else  { return "" }
        let nsRange = NSRange(location: fromIndex, length: toIndex + 1 - fromIndex)
        guard let range = Range(nsRange, in: self) else { return "" }
        return String(self[range])
    }
}

实际上,您不需要扩展,因为您已经有了一个NSRange,只需编写

代码语言:javascript
复制
func textView(_ textView: UITextView, shouldChangeTextIn range: NSRange, replacementText text: String) -> Bool {
    defer{
        if text.isEmpty {
            let textViewString = textView.attributedText.string
            guard let swiftRange = Range(range, in: textViewString) else { return }
            let deletedText = String(textViewString[swiftRange])
        }
    }
}
票数 0
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/58353002

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档