Decimal 类如何不可变?

qua*_*ine 5 python metaclass decimal immutability standard-library

我在使用 python3k 解释器时遇到了decimal.py 模块中的 Decimal 类。该类的文档表明从该类创建的对象是不可变的。我尝试查看 python 库代码,以了解程序员如何对类进行编码以使其不可变。不幸的是,我无法确切地发现他们做了什么(主要是因为班级本身相当大)。

我很想知道是否有人理解这个类如何实现其不可变状态。我认为这可能与重写类中的getattr () 和setattr () 方法有关,但我找不到任何证据。

我还注意到这个类能够以某种方式隐藏它在其插槽变量中定义的属性。也许该类使用某种类型的元类技术来实现这一点及其不变性。

如果有人有任何想法,我将不胜感激您的反馈。谢谢

mor*_*tar 3

好问题。对我来说,这似乎并不是不可改变的。Python 2.7.2:

>>> from decimal import Decimal
>>> d = Decimal('1.23')
>>> d
Decimal('1.23')
>>> f = d
>>> f
Decimal('1.23')
>>> d._exp = 1
>>> f
Decimal('1.23E+3')
Run Code Online (Sandbox Code Playgroud)

我确实看到文档说它是不可变的。我想他们的意思是如果您使用记录的接口,它是不可变的。例如

>>> d += 100
>>> f
Decimal('1.23E+3')
>>> d
Decimal('1330')
Run Code Online (Sandbox Code Playgroud)

其工作原理很简单,Decimal 没有?定义运算符重载。= 运算符:类似__i[op]__. 在这种情况下,d += 100转换为d = d.__add__(100),它返回一个新对象并更改 d 的标识,同时不影响原始对象。

  • 在Python 3.3中,decimal模块是用C实现的,并且默认使用它。纯 Python 实现仍然存在,但通常不使用。C 版本明显更快,但一些实现细节发生了变化。_exp 是一种不受支持、使用风险自负的方法,并且仅在纯 Python 实现中可用。它的使用或滥用可能会导致意想不到的结果。 (4认同)