我刚刚将
Xcode更新到8.0 beta 2和swift 3.0.从swift 2.3更新后,我遇到了很多错误.
我有一个String-extension,它将“self”-string中的Range转换为NSRange:
extension String { func NSRangeFromrange(_ range : Range<String.Index>) -> NSRange { let utf16view = self.utf16 let from = String.UTF16View.Index(range.lowerBound,within: utf16view) let to = String.UTF16View.Index(range.upperBound,within: utf16view) print("to: \(to) from: \(from)") print(self.characters.count) return NSMakeRange(utf16view.startIndex.distance(to: from),from.distance(to: to)) // return NSMakeRange(0,0) // <-- removes exception } }
Terminating app due to uncaught exception ‘NSRangeException’,reason:
‘NSMutableRLEArray objectAtIndex:effectiveRange:: Out of bounds’
当我打印到索引和索引时,我得到:
to: Index(_offset: 194) from: Index(_offset: 181)
字符串的字符数是210,这似乎是正确的.
所以,我不明白为什么它告诉我索引是超出界限的,当它们少于总数.
在我更新到swift 3之前,这条线路工作正常.当时它看起来像这样:
return NSMakeRange(utf16view.startIndex.distanceto(from),from.distanceto(to))
自动转换器没有将语法从swift 2.3更新到3.0,我自己做了..
有线索吗?
解决方法
在Swift 3中,“集合移动他们的索引”,请参阅
关于Swift进化的 A New Model for Collections and Indices.
关于Swift进化的 A New Model for Collections and Indices.
在Swift 2.2中,advancedBy()和distanceto()方法是
调用索引:
let u16 = "12345".utf16 let i1 = u16.startIndex.advancedBy(1) let i2 = u16.startIndex.advancedBy(3) let d = i1.distanceto(i2) print(d) // 2
这种方法仍然存在于Swift 3中,但至少在字符集上会产生意想不到的结果:
let u16 = "12345".utf16 let i1 = u16.startIndex.advanced(by: 1) let i2 = u16.startIndex.advanced(by: 3) let d = i1.distance(to: i2) print(d) // -2
正确的方法是使用index()和distance()方法
集合本身:
let u16 = "12345".utf16 let i1 = u16.index(u16.startIndex,offsetBy: 1) let i2 = u16.index(u16.startIndex,offsetBy: 3) let d = u16.distance(from: i1,to: i2) print(d) // 2
适用于您的问题,这是您的方法
转换范围< String.Index>在Swift 3中对应的NSRange
(从https://stackoverflow.com/a/30404532/1187415复制):
extension String { func nsRange(from range: Range<String.Index>) -> NSRange { let utf16view = self.utf16 let from = range.lowerBound.samePosition(in: utf16view) let to = range.upperBound.samePosition(in: utf16view) return NSMakeRange(utf16view.distance(from: utf16view.startIndex,to: from),utf16view.distance(from: from,to: to)) } }
而且为了完整起见,这是相反的转换
extension String { func range(from nsRange: NSRange) -> Range<String.Index>? { guard let from16 = utf16.index(utf16.startIndex,offsetBy: nsRange.location,limitedBy: utf16.endindex),let to16 = utf16.index(from16,offsetBy: nsRange.length,let from = String.Index(from16,within: self),let to = String.Index(to16,within: self) else { return nil } return from ..< to } }
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 [email protected] 举报,一经查实,本站将立刻删除。