如何使用Lua将一对字节转换为带符号的16位整数?
2016-12-3 12:45:17
收藏:0
阅读:132
评论:3
我正在编写一个需要将小的(最多16位)有符号整数序列化为二进制文件的程序。作为其中的一部分,我需要编写一个函数,将一个整数拆分为两个字节,另一个函数将这对字节转换回原始整数。
我脑海中出现的第一个想法是通过类似C语言的方式解决这个问题:
function dump_i16(n)
assert (-0x8000 <= n and n < 0x8000) -- 16 bit
local b1 = (n >> 8) & 0xff
local b2 = (n >> 0) & 0xff
return b1, b2
end
function read_i16(b1, b2)
assert (0 <= b1 and b1 <= 0xff) -- 8 bit
assert (0 <= b2 and b2 <= 0xff) -- 8 bit
return (b1 << 8) | (b2 << 0)
end
但是,这些函数对负数无效,因为Lua整数实际上有64位,我只保留了低16位:
-- 正数没问题
print( read_i16(dump_i16(17)) ) -- 17
print( read_i16(dump_i16(42)) ) -- 42
-- 负数返回不正确
print( read_i16(dump_i16(-1)) ) -- 65535 = 2^16 - 1
print( read_i16(dump_i16(-20)) ) -- 65516 = 2^16 - 20
什么是修改 read_i16 函数使其能够正确处理负数的最简洁方式?
如果可能的话,我希望使用纯Lua 5.3来完成这个任务,而不是编写C代码。
点赞
用户1847592
Lua 5.3 有特殊的转换函数。
要将一个整数-32768..32767 转换为长度为2的字符串,采用大端序:
local a_string = string.pack(">i2", your_integer)
要将其转换回来(将“a_string”的前两个字节转换):
local an_integer = string.unpack(">i2", a_string)
2016-12-03 07:50:58
用户90511
一种修复read_i16的方法是使用内部由string.pack使用的符号扩展算法:
function read_i16(b1, b2)
assert (0 <= b1 and b1 <= 0xff)
assert (0 <= b2 and b2 <= 0xff)
local mask = (1 << 15)
local res = (b1 << 8) | (b2 << 0)
return (res ~ mask) - mask
end
2016-12-03 13:31:29
评论区的留言会收到邮件通知哦~
推荐文章
- Lua 虚拟机加密load(string.dump(function)) 后执行失败问题如何解决
- 我想创建一个 Nginx 规则,禁止访问
- 如何将两个不同的lua文件合成一个 东西有点长 大佬请耐心看完 我是小白研究几天了都没搞定
- 如何在roblox studio中1:1导入真实世界的地形?
- 求解,lua_resume的第二次调用继续执行协程问题。
- 【上海普陀区】内向猫网络招募【Skynet游戏框架Lua后端程序员】
- SF爱好求教:如何用lua实现游戏内调用数据库函数实现账号密码注册?
- Lua实现网站后台开发
- LUA错误显式返回,社区常见的规约是怎么样的
- lua5.3下载库失败
- 请问如何实现文本框内容和某个网页搜索框内容连接,并把网页输出来的结果反馈到另外一个文本框上
- lua lanes多线程使用
- 一个kv数据库
- openresty 有没有比较轻量的 docker 镜像
- 想问一下,有大佬用过luacurl吗
- 在Lua执行过程中使用Load函数出现问题
- 为什么 neovim 里没有显示一些特殊字符?
- Lua比较两个表的值(不考虑键的顺序)
- 有个lua简单的项目,外包,有意者加微信 liuheng600456详谈,最好在成都
- 如何在 Visual Studio 2022 中运行 Lua 代码?

你需要使用逻辑右移来处理符号。参见>>>和>>之间的区别。Lua没有它,所以你需要自己实现。由于Lua整数默认为64位,因此你需要屏蔽掉64减去你所做的移位次数。
function dump_i16(n) assert (-0x8000 <= n and n < 0x8000) local b1 = (n >> 8) & ~(-1<<(64-8)) local b2 = (n >> 0) & 0xff return b1, b2 end