does object key-sharing happen when attributes are declared in a function that is called by __init__?
是的,无论您在何处设置属性,授予初始化后都具有相同的密钥集,实例字典使用共享密钥字典实现.提出的两种情况都减少了内存占用.
您可以使用sys.getsizeof来获取实例字典的大小,然后将其与从中创建的类似字典进行比较. dict .__ sizeof __的实现基于此判别返回不同的大小:
# on 64bit version of Python 3.6.1
print(sys.getsizeof(vars(c)))
112
print(getsizeof(dict(vars(c))))
240
所以,要找出答案,你需要做的就是比较这些.
至于你的编辑:
“If a single key is added that is not in the prototypical set of keys, you loose the key sharing”
正确,这是我(目前)发现的破坏共享密钥用法的两件事之一:
>在实例dict中使用非字符串键.这只能以愚蠢的方式进行. (你可以使用vars(inst).update)
>同一类的两个实例的字典内容偏离,这可以通过更改实例字典来完成. (添加到其中的单个键不在原型键集中)
我不确定在添加单个密钥时是否会发生这种情况,这是一个可能会发生变化的实现细节. (附录:见Martijn的评论)
这两件事都会导致CPython使用“普通”字典.当然,这是一个不应该依赖的实现细节.您可能会或可能不会在Python的其他实现和/或CPython的未来版本中找到它.