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