Python3序列化和反序列化实现

一、说明

早上看到Python使用pickle进行序列化和反序列化,然后发现面临的一个获取不到返回值的框架,似乎可以通过在框架中先序列化,然后在外部进行反序列化的方法来实现。就研究了一下pickle库的具体使用。

本身也没什么复杂,一方面还是怕忘记,另一方面是自从学Java听到反序化这个词开始就有一种莫明其妙的恐具感总觉得是什么高大上的东西。Java反序列化可参见“Java反序列化漏洞实现”。

 

二、本地的序列化与反序列化实现

import pickle


class BeSerializing:
    def __init__(self):
        self.var = "a test string."

    def print_var(self):
        print(f"self var value is:{self.var}")


class Serializing:
    def do_serializing(self):
        obj = BeSerializing()
        # with的好处是会最后自动释放获取的资源,对于文件就是最后会自己close()
        with open("test_pickle.obj", "wb") as file_handler:
            # 序列化,将对象写到文件
            pickle.dump(obj, file_handler)


class DeSerializing:
    def do_serializing(self):
        with open("test_pickle.obj", "rb") as file_handler:
            # 反序列化,将对象从文件中还原
            # 注意,虽然python的变量不需要使用前先定义类型,但使用时是要被格式化成确定类型的
            # 所以,如果反序列化类和被序列化类不在同一文件中,那么需要将被序列化类import进来,不然会因找不到被反序列化类而反序列化失败
            obj = pickle.load(file_handler)
            print(f"obj type: {type(obj)}\n"
                  f"obj.var value: {obj.var}")


if __name__ == "__main__":
    obj_se = Serializing()
    obj_se.do_serializing()
    obj_de = DeSerializing()
    obj_de.do_serializing()

运行结果如下,可以看到成功反序列化并打印出被序列化的对象的变量的值:

 

三、网络的序列化与反序列化实现

上边我们是以文件作为中介实现序列化与反序列化的,但如果到网络中就没有文件作为中介。此时可以使用pickle.dumps()和pickle.loads()进行序列化与反序列化。

import binascii
import pickle


class BeSerializing:
    def __init__(self):
        self.var = "a test string."

    def print_var(self):
        print(f"self var value is:{self.var}")


class NoFileSerializing:
    def do_serializing(self):
        obj_src = BeSerializing()
        # 二进制直接decode可能很多位置没法解码(codec can't decode)
        obj_src_bytes = pickle.dumps(obj_src)
        # 所以先转成ascii码形式的16进制再decode
        obj_src_str = binascii.b2a_hex(obj_src_bytes).decode(encoding="ascii")

        # 模拟网络远端获取传过来的字符串
        obj_dst_str = obj_src_str
        # 还原成二进制
        obj_dst_bytes = binascii.a2b_hex(obj_dst_str.encode(encoding="ascii"))
        # 反序列化
        obj_dst = pickle.loads(obj_dst_bytes)
        print(obj_dst.var)


if __name__ == "__main__":
    obj_nf = NoFileSerializing()
    obj_nf.do_serializing()

执行结果如下:

 

参考:

https://docs.python.org/3/library/pickle.html

posted on 2020-03-27 18:22  诸子流  阅读(1683)  评论(0编辑  收藏  举报