在用C/C++写网络打包/解包时,都是自己用指针偏移来实现对象<->二进制之间的互转,现在既然用了python,也要用一下序列化的思想,所以就使用cPickle来完成这件事情。
不过事情并不顺利……
假设我们在server端和client端通过类MManRsp来进行通信,这个类定义在mmanpro.py中,代码如下:
class MManRsp:
ret = 0
pathverlist = []
现在server端对client端回包如下:(源代码并非如此,为了演示,所以简化了代码)
rsp = MManRsp()
data = [('1',1),('2',2)]
for t in data:
x = do_something(t)#还是返回('1',1)这种类型
rsp.pathverlist.append(x)
senddata = cPickle.dumps(rsp)
#send
client端解析为:
rsp = cPickle.loads(recvdata)
print rsp.pathverlist
在运行这段代码的时候,client的print结果会是 []。
在经过漫长的原因查找之后终于知道,原来cPickle在序列化类实例时,对类内部变量是用引用的方式储存的,那么杯具的事情就来了,在server端,cPickle把引用序列化到了流中,而在client端根本就不不认识这个引用……(我浅显的理解,不对请大家千万指明)
所以需要在server端做如下修改:
rsp = MManRsp()
data = [('1',1),('2',2)]
newdata = []
for t in data:
x = do_something(t)#还是返回('1',1)这种类型
newdata.append(x)
rsp.pathverlist = copy.deepcopy(newdata)
senddata = cPickle.dumps(rsp)
这样就正常了。
但是还是剩下两点比较困惑:
1.rsp.pathverlist = copy.deepcopy(newdata),如果改成 rsp.pathverlist = newdata 也可以解决问题,不是说所有的赋值都引用么,奇怪……
2.在client端loads的时候,cPickle会自动找寻mmanpro.MManRsp这个模块,都不用在client的代码中加入from mmanpro import MManRsp
对python的了解还是不深,希望有知道原因的朋友能够不吝赐教。
另:
在网上看到一个关于cPickle的文档,写的很好。
http://www.ibm.com/developerworks/cn/linux/l-pypers/
评论
暂无评论