9

如果我写:

d = { 0:'a', 1:'b' }
d[False] = 'c'
d[True] = 'd'
print(d)

我得到:

{ 0:'c', 1:'d' }

为什么将它们强制转换为整数?它反过来做同样的事情。

d = {False:'a', True:'b'}
d[0] = 'c'
d[1] = 'd'
print(d)

输出是:

{False:'c', True: 'd'}

这可以禁用吗?它是一个错误吗?

4

3 回答 3

6

只是提供一些关于 arshajii 答案的背景。

两个布尔值TrueFalse与整数有着奇怪的关系。

一方面,它们具有不同的字符串表示形式,并且具有不同的身份:

>>> print(True)
True
>>> print(1)
1

>>> True is 1
False

另一方面,它们在比较和算术中表现为整数:

>>> True == 1
True
>>> True + 1
2

这种行为的原因是兼容性。很久以前,这种bool类型不存在。“布尔”运算符复制了 C 的行为,重用了“假”01“真”。

最终 Guido 意识到这没有多大意义,并添加了我们知道和喜爱的常量。

但有一个问题。即便如此,已经有很多代码将布尔值处理为整数。如果布尔运算开始使用“正确”类型,那么所有这些代码都会中断。

所以Guido做出了妥协。布尔值有自己的类型,bool,并且显示方式与整数不同。但是在算术运算和比较中,最值得注意的是__eq____hash__,它们被视为一回事。所以旧代码将继续工作,而新代码仍然可以利用新bool类型。

也许这会在 Python 4 中改变。但现在,bool是 的子类int,我们将不得不忍受它。

(在相关的说明中,这就是为什么TrueandFalse是 Title Case,而不是像其他 Python 关键字那样小写的原因之一。)

于 2013-11-08T04:00:35.773 回答
6

这是因为这些值被认为是相等的:

>>> True == 1
True
>>> 
>>> False == 0
True

并具有相同的哈希值:

>>> hash(True), hash(1)
(1, 1)
>>> 
>>> hash(False), hash(0)
(0, 0)

因此,从字典的角度来看,True1是无法区分False0

没有办法“禁用”它——你不应该在字典中使用非同质键。

在这种特定情况下,一种潜在的解决方法是分别为TrueFalse以外的1和保留特殊的 int 值0(假设您需要10作为独立键)。例如,您可以使用-1代表True-2代表False

于 2013-11-08T04:00:58.560 回答
1

boolint表示为“False”或“True”的子类,但其值为0 或 1。

于 2013-11-08T04:01:18.527 回答