gpt4 book ai didi

ios - String.subscript.getter 的 NSRangeException

转载 作者:行者123 更新时间:2023-11-28 11:25:35 24 4
gpt4 key购买 nike

我在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/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com