微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

斯威夫特3;范围’越界’

我刚刚将 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
    }
}

当执行NSMakeRange时,我收到一个错误

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 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] 举报,一经查实,本站将立刻删除。

相关推荐