- 最近在研究luasocket,准备用全部用lua的扩展库来实现一套轻量级框架,用来做一些工具、简单的游戏服务器,以及作为网络库用在cocos2dx中。
- 完善的网络库必然会遇到粘包、半包的问题,luasocket也不例外,由于网络部分在lua,协议的制定和buff的解析都没有合适的方案,又不想在C 中来封装接口,后面在网上查了一些资料,发现lua也有一个二进制打包的扩展库——lpack,了解之后发现还是蛮好用的,就决定使用它来做buff解析,用以解决粘包、半包的问题。
- 首先需要下载lpack的源码,地址是lpack的下载地址,然后解压,里面有lpack.c和makefile文件。
- 然后是编译lpack,有两种方式来编译和初始化lpack 1、将lpack.c加到宿主程序的源码里面,然后在初始化lua的地方调用初始化函数: luaopen_pack(lua_state); 2、将lpack编译成dll(so),然后在lua里面调用: require("lpack")
- 然后说一下lpack的接口,lpack扩展库提供了两个接口,pack和unpack。但是有两种调用方法,使用宏(USE_GLOBALS)来控制,第一种是使用全局接口(pack/unpack),另外一种是将这两个接口插入到string的空间里面(string.pack/string.unpack)。
lpack的具体用法 1、打包接口pack的使用,全局名字容易混淆lua本身函数unpack,使用string.pack好些,也可以修改源码修改函数名。
代码语言:javascript复制--luapack为我修改的接口名字
local _ss = luapack(">P", "中国asd")
local _ss2 = luapack(">h", 500)
_ss = _ss.._ss2
print(_ss, #_ss)
--调用结果
中国asd 9
中国asd 11
2、解包接口unpack的使用
代码语言:javascript复制--luaunpack为我修改的接口名字
--使用方式1
local ne, value = luaunpack(_ss, ">P")
print(ne, value)
_ss = string.sub(_ss, ne, #_ss)
local ne, value = luaunpack(_ss, ">h")
print(ne, value)
--调用结果
10 中国asd
3 500
--使用方式2
local ne, value1, value2 = luaunpack(_ss, ">Ph")
print(ne, value, value2)
--调用结果
12 中国asd 500
打包变量类型定义
代码语言:javascript复制
#define OP_ZSTRING 'z' //空字符串
#define OP_BSTRING 'p' //长度小于2^8的字符串
#define OP_WSTRING 'P' //长度小于2^16的字符串
#define OP_SSTRING 'a' //长度小于2^32/64的字符串*/
#define OP_STRING 'A' //指定长度字符串
#define OP_FLOAT 'f' /* float */
#define OP_DOUBLE 'd' /* double */
#define OP_NUMBER 'n' /* Lua number */
#define OP_CHAR 'c' /* char */
#define OP_BYTE 'b' /* byte = unsigned char */
#define OP_SHORT 'h' /* short */
#define OP_USHORT 'H' /* unsigned short */
#define OP_INT 'i' /* int */
#define OP_UINT 'I' /* unsigned int */
#define OP_LONG 'l' /* long */
#define OP_ULONG 'L' /* unsigned long */
打包字节序,分为以下3种
代码语言:javascript复制 #define OP_LITTLEENDIAN '<' /* little endian */
#define OP_BIGENDIAN '>' /* big endian */
#define OP_NATIVE '=' /* native endian */
- 好了,到这里如何使用lpack已经很清楚,接着就可以利用lpack的特性设计buff来解析网络字节流了。再次吐槽一下,这个编辑器没换缩进很难使用啊!!