1

我正在尝试使用如下代码[UInt8]的功能进行扩展。getUInt32BE()

这是我收到的错误:

“不能用 'Range' 类型的索引下标 'Self' 类型的值”

有人可以帮我纠正这个错误吗?

extension Collection where Iterator.Element == UInt8 {
    public func getUInt32BE(at: Int = 0) -> UInt32 {
        return self[at..<at+4].reduce(0) {
          $0 << 8 + UInt32($1)
        }
    }
}

提前致谢 :)

4

1 回答 1

1

备选方案#1:使用.pointee的属性UnsafePointer

如以下问答中所述

您可以在 Swift 2.2 中使用,例如

UnsafePointer<UInt16>(bytes).memory

将 2 字节UInt8数组转换为UInt16值(主机字节顺序)。

现在,在 Swift 3.0-dev.memory中已经被.pointee. 应用上述以及此更改,我们同样可以使用UnsafePointer<UInt32>(bytes).pointee来访问UInt324 字节UInt8数组的表示,但请注意该表示使用主机字节顺序进行转换。为了可能(如有必要)将其转换为整数的大端表示,我们可以使用.bigEndian每个整数类型可用的属性,如下面的问答中所述:

这在 Swift 3.0-dev 中仍然有效,因此,我们可以getUInt32BE(...)如下构造您的方法

extension Collection where Iterator.Element == UInt8 {
    public func getUInt32BE(at: Index.Distance) -> UInt32? {
        let from = startIndex.advanced(by: at, limit: endIndex)
        let to = from.advanced(by: 4, limit: endIndex)
        
        guard case let bytes = Array(self[from..<to]) 
            where bytes.count == 4 else { return nil }
        
        return UnsafePointer<UInt32>(bytes).pointee.bigEndian
    }
}

备选方案#2:使用位移

或者,使用带有位移的上述修改版本(类似于您的问题的尝试)而不是使用UnsafePointer<..>

extension Collection where Iterator.Element == UInt8 {

    public func getUInt32BE(at: Index.Distance) -> UInt32? {
        let from = startIndex.advanced(by: at, limit: endIndex)
        let to = from.advanced(by: 4, limit: endIndex)
        
        guard case let bytesSlice = self[from..<to] 
            where from.distance(to: to) == 4 else { return nil }
        
        return bytesSlice.reduce(0) { (tot, val) -> UInt32 in
            tot << 8 + UInt32(val as! UInt8)
        }
    }
}

val请注意,我们需要通过强制转换from Iterator.Elementto来帮助编译器(由于扩展的子句,UInt8这保证成功; )。whereIterator.Element == UInt8


示例用法

上述两种选择中的任何一种的示例用法:

/* example usage */
let bytes: [UInt8] = [
    0,   // 0b 0000 0000
    255, // 0b 1111 1111 
    0,   // 0b 0000 0000
    104, // 0b 0110 1000
    76]  // 0b 0100 1100
    
/*  byteArr[1..<5], big-endian:
        0b 1111 1111 0000 0000 0110 1000 0100 1100 */
let u32verify = 0b11111111000000000110100001001100 
print(u32verify) // 4278216780
    
if let u32val = bytes.getUInt32BE(1) {
    print(u32val) // 4278216780, OK
}

有关Collection协议的详细信息(上述错误的原因是未使用Index.Distance此协议的关联类型来构造您的子数组),请参见例如

于 2016-04-26T21:48:42.530 回答