Python 使用 __getstate__ 和 __setstate__ 魔法方法
发布日期:2021-05-06 19:34:31 浏览次数:10 分类:技术文章

本文共 1943 字,大约阅读时间需要 6 分钟。

__getstate__ 与 __setstate__ 两个魔法方法分别用于Python 对象的序列化与反序列化

在序列化时, _getstate__ 可以指定将那些信息记录下来, 而 __setstate__ 指明如何利用已记录的信息

先创建一个基类

class Demo:    def __init__(self, name, age=0):        self.name = name        self.age = age    def __str__(self):        return f"name: {self.name}\nage: {self.age}"

继承Demo, 并实现 __getstate__ 与 __setstate__ 方法

class DemoState1(Demo):    # 反序列化时调用, state 是 __getstate__ 的返回对象    def __setstate__(self, state):        print("invoke __setstate__")        # 将记录的信息("Python3") 赋给 name        self.name = state        self.age = 31    # 序列化时调用    def __getstate__(self):        print("invoke __getstate__")        # 记录信息 “Python3”        return "Python3"

测试调用流程

In [3]: import pickleIn [4]: demo_1 = DemoState1("None") # "None" 0In [5]: print(demo_1)name: Noneage: 0In [6]: demo_bytes_1 = pickle.dumps(demo_1) # 序列化为字节串invoke __getstate__In [7]: print(demo_bytes_1) # 仅对象中的字符串信息是可读的b'\x80\x03c__main__\nDemoState1\nq\x00)\x81q\x01X\x07\x00\x00\x00Python3q\x02b.'In [8]: demo_object_1 = pickle.loads(demo_bytes_1)invoke __setstate__In [9]: print(demo_object_1)name: Python3age: 31

可以省略 __setstate__, 但 __getstate__ 必须返回一个字典, 字典的内容被加载到当前的对象

class DemoState2(Demo):    # 省略 __setstate__    # 自动将 __getstate__ 的返回对象添加到当前对象的属性    # 序列化时调用    def __getstate__(self):        # 必须返回字典        state = {   "name": "C", "age": 50, "address": "Bell Lab"}        return state

测试

In [11]: demo_2 = DemoState2("None")In [12]: demo_bytes_2 = pickle.dumps(demo_2)In [13]: print(pickle.loads(demo_bytes_2))name: Cage: 50

对象本身没有 address 属性, 因此反序列化时被忽略了

如果 __getstate__ 与 __setstate__ 都省略, 那么就是默认情况, 自动保存和加载对象的属性字典 __dict__

In [14]: js = Demo("JavaScript", 26)In [15]: js.__dict__ # 对象的属性字典                                     Out[16]: {   'name': 'JavaScript', 'age': 26}In [17]: js_bytes = pickle.dumps(js)In [18]: print(pickle.loads(js_bytes))name: JavaScriptage: 26

大多数时候保持默认情况即可, 但是遇到不可序列化的对象, 例如 self.file = open("filename"), 就需要忽略掉该属性

上一篇:GitHub OAuth 第三方登录(使用 Flask)
下一篇:XML解析库 lxml 教程(三)

发表评论

最新留言

不错!
[***.144.177.141]2025年03月13日 08时44分17秒