技术头条 - 一个快速在微博传播文章的方式     搜索本站
您现在的位置首页 --> 其他 --> cPickle序列化自定义类实例时的陷阱

cPickle序列化自定义类实例时的陷阱

浏览:1728次  出处信息

在用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/

建议继续学习:

  1. JavaScript性能陷阱    (阅读:3139)
  2. Java陷阱(2010版)    (阅读:3194)
  3. PHP数据类型隐性转换的陷阱    (阅读:2950)
  4. 移动互联网系统架构十大陷阱    (阅读:2762)
  5. 类型转换-无处不在的陷阱    (阅读:2244)
  6. JavaScript 中的陷阱    (阅读:1817)
  7. 可能被你忽略的 JavaScript 代码陷阱    (阅读:1712)
  8. 情景反射陷阱    (阅读:1699)
QQ技术交流群:445447336,欢迎加入!
扫一扫订阅我的微信号:IT技术博客大学习
后一篇:Catalyst 框架学习 >>
© 2009 - 2024 by blogread.cn 微博:@IT技术博客大学习

京ICP备15002552号-1