简体   繁体   English

如何在 swift 中将字节转换为浮点值?

[英]How to convert bytes to a float value in swift?

This is my code to convert byte data to float.这是我将字节数据转换为浮点数的代码。 I tried every answers given in this site.我尝试了这个网站上给出的每一个答案。 I am getting exponential value for this "<44fa0000>" byte data我得到这个“<44fa0000>”字节数据的指数值

    static func returnFloatValue(mutableData:NSMutableData)->Float
    {
       let qtyRange = mutableData.subdataWithRange(NSMakeRange(0, 4))
       let qtyString = String(qtyRange)
       let qtyTrimString = qtyString.stringByTrimmingCharactersInSet(NSCharacterSet(charactersInString: "<>"))
       let qtyValue =  Float(strtoul(qtyTrimString, nil, 16)/10)
       return qtyValue
    }

Thanks谢谢

<44fa0000> is the big-endian memory representation of the binary floating point number 2000.0 . <44fa0000>是二进制浮点数2000.0的大端内存表示。 To get the number back from the data, you have to read it into an UInt32 first, convert from big-endian to host byteorder, and then cast the result to a Float .要从数据中取回数字,您必须首先将其读入UInt32 ,从 big-endian 转换为主机字节顺序,然后将结果转换为Float

In Swift 2 that would beSwift 2 中,这将是

func floatValueFromData(data: NSData) -> Float {
    return unsafeBitCast(UInt32(bigEndian: UnsafePointer(data.bytes).memory), Float.self)
}

Example:例子:

let bytes: [UInt8] =  [0x44, 0xFA, 0x00, 0x00]
let data = NSData(bytes: bytes, length: 4)

print(data) // <44fa0000>
let f = floatValueFromData(data)
print(f) // 2000.0

In Swift 3 you would use Data instead of NSData , and the unsafeBitCast can be replaced by the Float(bitPattern:) initializer:Swift 3 中,您将使用Data而不是NSData ,并且unsafeBitCast可以替换为Float(bitPattern:)初始值设定项:

func floatValue(data: Data) -> Float {
    return Float(bitPattern: UInt32(bigEndian: data.withUnsafeBytes { $0.pointee } ))
}

In Swift 5 the withUnsafeBytes() method of Data calls the closure with an (untyped) UnsafeRawBufferPointer , and you can load() the value from the raw memory:Swift 5中, DatawithUnsafeBytes()方法使用(无类型的) UnsafeRawBufferPointer调用闭包,您可以从原始内存中load()值:

func floatValue(data: Data) -> Float {
    return Float(bitPattern: UInt32(bigEndian: data.withUnsafeBytes { $0.load(as: UInt32.self) }))
}

Here is some swift 5 :这是一些swift 5

let data = Data([0x44, 0xfa, 0x00, 0x00]) // 0x44fa0000
let floatNb:Float = data.withUnsafeBytes { $0.load(as: Float.self) }

// note that depending on the input endianess, you could add .reversed() to data
let floatNb:Float = data.reversed().withUnsafeBytes { $0.load(as: Float.self) }

WARNING : this sample throws if your Data is under 4 bytes..警告:如果您的数据小于 4 字节,则此示例会抛出。

. .

Safe Data extension:安全数据扩展:

extension Data {
    enum Endianess {
        case little
        case big
    }

    func toFloat(endianess: Endianess = .little) -> Float? {
        guard self.count <= 4 else { return nil }

        switch endianess {
        case .big:
            let data = [UInt8](repeating: 0x00, count: 4-self.count) + self
            return data.withUnsafeBytes { $0.load(as: Float.self) }
        case .little:
            let data = self + [UInt8](repeating: 0x00, count: 4-self.count)
            return data.reversed().withUnsafeBytes { $0.load(as: Float.self) }
        }
    }
}

Tests:测试:

let opData        = Data([0x44, 0xFA, 0x00, 0x00])
let nb42          = Data([0x42, 0x28])
let nb42bigEndian = Data([0x28, 0x42])
let tooBig        = Data([0x44, 0xFA, 0x00, 0x00, 0x00])

print("opData:        \(opData.toFloat())")
print("nb42:          \(nb42.toFloat())")
print("nb42bigEndian: \(nb42bigEndian.toFloat(endianess: .big))")
print("tooBig:        \(tooBig.toFloat())")

you may find a faster way but this was good enough for my needs您可能会找到更快的方法,但这足以满足我的需求

Use this function:使用这个功能:

static func returnFloatValue(data: NSMutableData) -> Float {
    let bytes = [UInt8](data as Data)
    var f: Float = 0

    memcpy(&f, bytes, 4)
    return f
}

And you can see it in action here:你可以在这里看到它的实际效果:

var initialValue: Float = 19.200

let data = NSMutableData(bytes: &initialValue, length: 4)

func returnFloatValue(data: NSMutableData) -> Float {
    let bytes = [UInt8](data as Data)
    var f: Float = 0

    memcpy(&f, bytes, 4)
    return f
}

var result:Float = returnFloatValue(data: data)

print("f=\(result)")// f=19.2

For 64 bit values the code is:对于 64 位值,代码为:

static func longBitsToDouble(x : Int64) -> Float64
{ return Float64(bitPattern: UInt64(x)) } 

Swift 4+. Swift 4+。

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM