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

python单进程能否利用多核cpu的测试结论

在很早的时候,就听网上的文章说: python有GIL,所以在单进程内,即使使用多线程也无法利用到多核的优势,同一时刻,python的字节码只会运行在一个cpu上。 ...

阅读全文

python常用web框架性能测试(django,flask,bottle,tornado)

测了一下django、flask、bottle、tornado 框架本身最简单的性能。对django的性能完全无语了。 django、flask、bottle 均使用gunicorn+gevent启动...

阅读全文

手机应用/服务器开发的一些总结(一)

经过这么久在android客户端和服务器端的开发,感觉还是积累了不少东西想要和大家分享一下,但是好想单独拎一个点出来又不太值得,所以就汇集到一起写成系列吧...

阅读全文

发表评论