- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我想问“如何 pickle 一个继承自 dict
并定义 __slots__
的类”。然后我在 class B
中意识到了完全令人痛苦的解决方案下面实际上有效...
import pickle
class A(dict):
__slots__ = ["porridge"]
def __init__(self, porridge): self.porridge = porridge
class B(A):
__slots__ = ["porridge"]
def __getstate__(self):
# Returning the very item being pickled in 'self'??
return self, self.porridge
def __setstate__(self, state):
print "__setstate__(%s) type(%s, %s)" % (state, type(state[0]),
type(state[1]))
self.update(state[0])
self.porridge = state[1]
>>> saved = pickle.dumps(A(10))
TypeError: a class that defines __slots__ without defining __getstate__ cannot be pickled
>>> b = B('delicious')
>>> b['butter'] = 'yes please'
>>> loaded = pickle.loads(pickle.dumps(b))
__setstate__(({'butter': 'yes please'}, 'delicious')) type(<class '__main__.B'>, <type 'str'>)
>>> b
{'butter': 'yes please'}
>>> b.porridge
'delicious'
pickle
不能 pickle 定义
__slots__
的类没有同时定义
__getstate__
.如果类继承自
dict
,这是一个问题- 因为不返回
self
怎么返回实例的内容,这是pickle 已经尝试pickle 的实例,如果不调用
__getstate__
就不能这样做。 .注意如何
__setstate__
实际上正在接收一个实例
B
作为国家的一部分。
最佳答案
也许我参加聚会有点晚了,但是这个问题没有得到真正解释正在发生的事情的答案,所以我们开始吧。
对于那些不想阅读整篇文章的人来说,这是一个快速摘要(它有点长......):
dict
实例在 __getstate__()
-- pickle
会为你做这件事。 self
反正在状态,pickle
的循环检测将防止无限循环。 __getstate__()
和
__setstate__()
派生自
dict
的自定义类的方法
__getstate__()
和
__setstate__()
你的类的方法。您无需处理
dict
的内容的 pickle 。实例包含在
B
中实例 --
pickle
知道如何处理字典,并会为你做这件事。所以这个实现就足够了:
class B(A):
__slots__ = ["porridge"]
def __getstate__(self):
return self.porridge
def __setstate__(self, state):
self.porridge = state
>>> a = B("oats")
>>> a[42] = "answer"
>>> b = pickle.loads(pickle.dumps(a))
>>> b
{42: 'answer'}
>>> b.porridge
'oats'
pickle
模块遇到用户定义类的实例,它调用
__reduce__()
此类的方法,依次调用
__getstate__()
(实际上,它通常会调用
__reduce_ex__()
方法,但这在这里无关紧要)。让我们定义
B
再次像您最初所做的那样,即使用
__getstate__()
的“递归”定义,让我们看看调用
__reduce__()
会得到什么对于
B
的实例现在:
>>> a = B("oats")
>>> a[42] = "answer"
>>> a.__reduce__()
(<function _reconstructor at 0xb7478454>,
(<class '__main__.B'>, <type 'dict'>, {42: 'answer'}),
({42: 'answer'}, 'oats'))
__reduce__()
可以看出,该方法返回一个包含 2 到 5 个元素的元组。第一个元素是一个函数,在 unpickling 时将被调用以重构实例,第二个元素是将传递给该函数的参数元组,第三个元素是
__getstate__()
的返回值。 .我们已经可以看到字典信息被包含了两次。函数
_reconstructor()
是
copy_reg
的内部函数重建基类之前的模块
__setstate__()
在 unpickling 时调用。 (如果您愿意,可以查看
source code of this function - 它很短!)
a.__reduce__()
的返回值进行pickle .它基本上一个接一个地 pickle 这个元组的三个元素。第二个元素又是一个元组,它的元素也是一个接一个的pickle。此内部元组的第三项(即
a.__reduce__()[1][2]
)的类型为
dict
并使用字典的内部pickler 进行pickle。外部元组的第三个元素(即
a.__reduce__()[2]
)也是一个元组,由
B
组成实例本身和一个字符串。 pickle 时
B
例如,
cycle detection of the pickle
module开始:
pickle
意识到这个确切的实例已经被处理,并且只存储对其
id()
的引用。而不是真正 pickle 它——这就是为什么没有无限循环发生的原因。
B
字典部分已经初始化的实例。接下来,unpickler 读取状态。它遇到了一个元组,它包含对一个已经解压的对象的引用——即我们的
B
实例。 -- 和一个字符串,
"oats"
.这个元组现在传递给
B.__setstate__()
.
state
的第一个元素和
self
现在是同一个对象,通过添加行可以看出
print self is state[0]
__setstate__()
实现(它打印
True
!)。线
self.update(state[0])
关于python - 为什么pickle __getstate__ 接受它首先需要__getstate__ 进行pickle 的实例作为返回值?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/5247250/
我正在本地编程应用程序,但是当我迁移到服务器时,使用此行出现解析错误: if(!is_array($data[array_keys($data)[0]])) 返回值: Parse error: syn
关闭。这个问题是opinion-based 。目前不接受答案。 想要改进这个问题吗?更新问题,以便 editing this post 可以用事实和引文来回答它。 . 已关闭 9 年前。 Improv
我已经开始尝试用 C++ 学习 Winsock,但我遇到了一些问题。我首先遵循 MSDN 上的 Winsock 示例(一个基本的客户端-服务器)。 http://msdn.microsoft.com/
我有一段使用 epoll 的代码,但它有问题。当我运行它时,它给出输出:服务器套接字()没问题......服务器绑定(bind)()没问题......3个4个接受:无效参数 我在 ubuntu lin
我正在寻找一种方法来接受 $_GET 变量作为文件路径,例如 - /page/test将转换为 page.php?page=test .我已经搜索过,但找不到任何可以帮助我的东西,因此我在这里问。 我
我想要一个只接受从 0 到 9 和减号的浮点数的正则表达式。 请帮忙。 最佳答案 ^[-+]?[0-9]*\.?[0-9]+$ ^ - 字符串开头 [-+]? - 0 或 1 符号指示符 [0-9]*
请问如何接受\r\n无需将其更改为 \\r\\n , 与 fgets . 我想让程序翻译 \r\n到换行符而不是将其打印为字符串。 当前代码: char buff[1024]; printf("Msg
我正在编写一个 Cocoa 应用程序,该应用程序需要在其 Dock 图标上接受已安装卷的滴落。它不是基于文档的;我打算将每个卷分派(dispatch)到 application:openFiles 中
我在 SQLite 中发现了这种意外行为。 SQLite 似乎接受 SQL 连接语法中的任意关键字。如果我不小心键入了 natural join 而不是 natural join,则会生成笛卡尔积。这
我在 Windows 窗体的同一个窗体上有一个 TreeView 和一个多行文本框。我有拖放设置,以便我可以将节点从 TreeView 拖到文本框并将文本插入文本框(这是有效的)。 我想增强这一点,以
我正在创建一棵类似于 D3 Layout Tree 的树并尝试绑定(bind)我的自定义 JSON 对象。切换树节点的代码如下。 function toggleAll(d) { if (d.c
所以,我希望能够向我的 DOB 字段发送空选项。 这是我的表单生成器: ->add('birthDate', DateType::class, array( 'widg
错误可以在这里看到:http://djaffry.selfip.com:8080/ 我希望索引页接受参数,无论是 mysite.com/search/param_here 或 mysite.com/?
我想知道标准 Scala 解析器组合器是否包含一个解析器,该解析器接受 Scala 语言本身也接受的相同标识符(如 Scala 语言规范第 1.1 节中所指定)。 StdTokenParsers 特征
我暂时使用以下行在我的 Android 手机上创建 ServerSocket: socketl = new ServerSocket(port, 0, InetAddress.getByName("1
今天早上刚刚发布了一个应用程序,我在所有可能的设备和模拟器上测试了它,但是当我从商店下载它时,应用程序在启动时崩溃。我在很多设备和iOS版本上测试过,但结果都是一样的: Incident Identi
我想要 5 个名字中最长的一个。我想我应该使用 compareTo() 方法或 length()? 输出必须是这样的: enter 5 names : Joey Mark Catherine Zach
抱歉,我不熟悉泛型,我可以创建一个如下所示的泛型类: public class InfoField { } 上面的类可以接受如下值: , User> 但是我可以有接受 map 对象作为值的类吗?
我想让一个单元格等于它上面的单元格。 当我写作时 =address(row()-1;column()) 它只是写入上面单元格的名称(例如,对于 B2,它写入 $B$1) 是否有一个函数可以输入地址并放
我正在使用Asm访问java字节码来获取方法信息。看来ClassReader类的accept方法是异步调用的,所以不可能立即获取方法信息。我该怎么办? 最佳答案 实际上,accept 方法是从您自己的
我是一名优秀的程序员,十分优秀!