6

我使用的是 Lua 的前端,不幸的是它已经过时了,所以我在这里坚持使用 5.1 版,这意味着bit32库无法访问(我可能可以用来转换它)。

所以我想知道是否有人知道我可以实现浮点到二进制(数字)函数或者更好的是浮点到十六进制的方法。到目前为止,我能想到的最好的方法是十进制到二进制/十六进制函数......

4

3 回答 3

3

以下函数使用 François Perrad 的一些代码lua-MessagePack。非常感谢他。

function float2hex (n)
    if n == 0.0 then return 0.0 end

    local sign = 0
    if n < 0.0 then
        sign = 0x80
        n = -n
    end

    local mant, expo = math.frexp(n)
    local hext = {}

    if mant ~= mant then
        hext[#hext+1] = string.char(0xFF, 0x88, 0x00, 0x00)

    elseif mant == math.huge or expo > 0x80 then
        if sign == 0 then
            hext[#hext+1] = string.char(0x7F, 0x80, 0x00, 0x00)
        else
            hext[#hext+1] = string.char(0xFF, 0x80, 0x00, 0x00)
        end

    elseif (mant == 0.0 and expo == 0) or expo < -0x7E then
        hext[#hext+1] = string.char(sign, 0x00, 0x00, 0x00)

    else
        expo = expo + 0x7E
        mant = (mant * 2.0 - 1.0) * math.ldexp(0.5, 24)
        hext[#hext+1] = string.char(sign + math.floor(expo / 0x2),
                                    (expo % 0x2) * 0x80 + math.floor(mant / 0x10000),
                                    math.floor(mant / 0x100) % 0x100,
                                    mant % 0x100)
    end

    return tonumber(string.gsub(table.concat(hext),"(.)",
                                function (c) return string.format("%02X%s",string.byte(c),"") end), 16)
end


function hex2float (c)
    if c == 0 then return 0.0 end
    local c = string.gsub(string.format("%X", c),"(..)",function (x) return string.char(tonumber(x, 16)) end)
    local b1,b2,b3,b4 = string.byte(c, 1, 4)
    local sign = b1 > 0x7F
    local expo = (b1 % 0x80) * 0x2 + math.floor(b2 / 0x80)
    local mant = ((b2 % 0x80) * 0x100 + b3) * 0x100 + b4

    if sign then
        sign = -1
    else
        sign = 1
    end

    local n

    if mant == 0 and expo == 0 then
        n = sign * 0.0
    elseif expo == 0xFF then
        if mant == 0 then
            n = sign * math.huge
        else
            n = 0.0/0.0
        end
    else
        n = sign * math.ldexp(1.0 + mant / 0x800000, expo - 0x7F)
    end

    return n
end
于 2013-11-15T08:43:43.537 回答
1

上例中的 float2hex 返回一个 int。话虽这么说,如果有人在这里需要它是一个 intToHex 转换函数,可以在 lua 档案(http://lua-users.org/lists/lua-l/2004-09/msg00054.html)中找到。我使用了上面 float2hex 函数的返回值并将其输入到这个函数中。intToHex 的输出是一个字符串(例如:0xA4CD)。

function intToHex(IN)
    local B,K,OUT,I,D=16,"0123456789ABCDEF","",0
    while IN>0 do
        I=I+1
        IN,D=math.floor(IN/B),math.mod(IN,B)+1
        OUT=string.sub(K,D,D)..OUT
    end


    OUT = "0x" .. OUT
    return OUT
end
于 2017-06-16T14:08:16.903 回答
1

我面临的问题是试图从 Hex 转换回 Float;您可以使用该 tonumber(x, 16) 将其与“错误 - 语法悔恨”响应结合使用。string.sub 只是因为我的 Modbus 驱动程序不交换字节:)

function convertFloatSwapWord(number) -- number is string of format = 0x########
     local x = 0
     x = float2hex(number)
     x = intToHex(x)
     x = string.sub(x,5,6) .. string.sub(x,3,4) .. string.sub(x,9,10) .. string.sub(x,7,8)
     x = hex2float(tonumber(x, 16))
     return x
end
于 2018-03-12T19:26:39.380 回答