最近用skynet,sproto通讯,完全看不懂通讯二进制是怎么写的,发现都是string这个,string那个,完全理解不来。
于是查了一下string.pack的api,和之前别的语言里面用的writeByte,writeShort,writeUnsignedInt这样的写法完全是不一样的。
然后在此记录下来。
参考
字符串的Pack和Unpack
API
string.pack, string.unpack, string.packsize的接收的第一个参数的一个格式化的字符串,这个字符串决定了接下来创建的字符串结构的读或者写的方式。
格式化字符串是一个约定序列,可用约定选项如下:
-
<: 使用little endian
-
>: 使用big endian
- =: 使用本地默认endian
- ![n]: 设置最大alignment为 n (默认本地 alignment)
- b: 一个带符号字节 (char)
- *B: 一个无符号字节 (char)
- h: 一个带符号短整(本地尺寸)
- H: 一个无符号短整(本地尺寸)
- l: 带符号长整型 (本地尺寸)
- L: 无符号长整型 (本地尺寸)
- j: 一个lua_Integer
- J: 一个lua_Unsigned
- T: 一个size_t (native size)
- i[n]: n个字节的带符号int (默认本地尺寸)
- I[n]: n个字节的无符号int (默认本地尺寸)
- f: 浮点型 (本地尺寸)
- d: double型 (本地尺寸)
- n: 一个lua_Number
- cn: 一个固定n个字节长度的字符串
- z: 一个 zero-terminated 字符串
- s[n]: 一个字符串,先将长度以无符号整型写入,后面是n个byte (默认是一个 size_t)
- x: 一个字节的 padding
- Xop: 一个根据他的选项操作排列的空白item (反之忽略 ignored)
- ' ': (空格) 忽略
除了padding,空格和configuration("xX <=>!")之外,每个选项都对应一个参数(string.pack)或者一个结果(string.unpack)
对于 !n , sn , in 以及 In ,_n_可以说1-16之间的任何整数。All integral options check overflows(所有整型都会检查溢出); string.pack 会检查提供的值是否适合提供的尺寸;string.unpack 会检查读取的值是否是一个 Lua Integer。【暂时不懂后半句】
如果字符串以 "!1=" 为前缀,with maximum alignment of 1 (no alignment) and native endianness. (以本地编码的1的最大排列)
排列的过程:对于每个选项,如果数据长度不够的话,都会以padding补齐。For each option, the format gets extra padding until the data starts at an offset that is a multiple of the minimum between the option size and the maximum alignment; this minimum must be a power of 2. "c" 和 "z" 不排列; "s" 遵循它开始的整型排列.
所有的 padding 在 string.pack时都是以0填充 (在 string.unpack 中忽略)
Demo
local str = string.pack(">i2",10)
print(str)
local str = string.unpack(">i2",str)
print(str)
output:
a
10
UTF8
居然有6.5 – UTF-8 Support
名词
|
请发表评论