如何在Python中获得布尔值的相反(否定)?


97

对于以下示例:

def fuctionName(int, bool):
    if int in range(...):
        if bool == True:
            return False
        else:
            return True

有什么办法可以跳过第二个if语句?只是为了告诉计算机返回布尔值的相反值bool


6
这可能仅仅是伪代码,但intbool都内置名(它们所代表的类型),并且不应该被用来作为变量名。
SingleNegationElimination

是的,这只是一个伪代码,仅用于演示目的……
amyassin 2011年

6
if x == True:应该写if x:
Mike Graham

Answers:


169

您可以使用:

return not bool

4
而且,int in range (....)效率低下。它将创建一个列表,然后执行线性搜索。优于x in range(low, high)low <= x < high
MRAB

48

not操作者(逻辑否定)

最好的方法可能是使用运算符not

>>> value = True
>>> not value
False

>>> value = False
>>> not value
True

因此,而不是您的代码:

if bool == True:
    return False
else:
    return True

您可以使用:

return not bool

逻辑否定为函数

operator模块中还有两个函数,如果需要将其作为函数而不是运算符,operator.not_则为别名operator.__not__

>>> import operator
>>> operator.not_(False)
True
>>> operator.not_(True)
False

如果您要使用需要谓词功能或回调的功能,这些功能将很有用。

例如mapfilter

>>> lst = [True, False, True, False]
>>> list(map(operator.not_, lst))
[False, True, False, True]

>>> lst = [True, False, True, False]
>>> list(filter(operator.not_, lst))
[False, False]

当然,使用等效lambda功能也可以实现相同的效果:

>>> my_not_function = lambda item: not item

>>> list(map(my_not_function, lst))
[False, True, False, True]

不要~在布尔值上使用按位求反运算符

一个人可能很想使用按位取反运算符~或等效的运算符功能operator.inv(或那里的其他三个别名之一)。但是由于结果bool的子类int可能是意外的,因为它不返回“反布尔”,而是返回“反整数”:

>>> ~True
-2
>>> ~False
-1

这是因为,True就相当于1False0和位反转上的按位表示操作整数 10

因此,这些不能用于“否定” a bool

NumPy数组(和子类)的求反

如果您要处理包含布尔值的NumPy数组(或子类,例如pandas.Seriespandas.DataFrame),则实际上可以使用按位逆运算符(~)来消除数组中的所有布尔值:

>>> import numpy as np
>>> arr = np.array([True, False, True, False])
>>> ~arr
array([False,  True, False,  True])

或等效的NumPy函数:

>>> np.bitwise_not(arr)
array([False,  True, False,  True])

您不能在NumPy数组上使用not运算符或operator.not函数,因为它们要求它们返回一个bool(而不是布尔数组),但是NumPy还包含一个逻辑非函数,可以在元素上起作用:

>>> np.logical_not(arr)
array([False,  True, False,  True])

这也可以应用于非布尔数组:

>>> arr = np.array([0, 1, 2, 0])
>>> np.logical_not(arr)
array([ True, False, False,  True])

自定义自己的班级

not通过调用bool值并取反结果来工作。在最简单的情况下,真值将仅调用__bool__对象。

因此,通过实现__bool__(或__nonzero__在Python 2中),您可以自定义true值,从而自定义结果not

class Test(object):
    def __init__(self, value):
        self._value = value

    def __bool__(self):
        print('__bool__ called on {!r}'.format(self))
        return bool(self._value)

    __nonzero__ = __bool__  # Python 2 compatibility

    def __repr__(self):
        return '{self.__class__.__name__}({self._value!r})'.format(self=self)

我添加了一条print语句,以便您可以验证它是否确实调用了该方法:

>>> a = Test(10)
>>> not a
__bool__ called on Test(10)
False

同样,您可以实现应用__invert__方法以在~应用时实现行为:

class Test(object):
    def __init__(self, value):
        self._value = value

    def __invert__(self):
        print('__invert__ called on {!r}'.format(self))
        return not self._value

    def __repr__(self):
        return '{self.__class__.__name__}({self._value!r})'.format(self=self)

再次print调用以查看它实际上是否被调用:

>>> a = Test(True)
>>> ~a
__invert__ called on Test(True)
False

>>> a = Test(False)
>>> ~a
__invert__ called on Test(False)
True

但是,这样的实现__invert__可能会造成混淆,因为它的行为不同于“常规” Python行为。如果您这样做的话,请清楚地记录下来,并确保它有一个很好的(和常见的)用例。


11

Python有一个“非”运​​算符,对吗?不仅仅是“不是”吗?如

  return not bool

3

您可以只比较布尔数组。例如

X = [True, False, True]

然后

Y = X == False

会给你

Y = [False, True, False]

1
也许对于Numpy数组,但是对于标准Python列表,这是不正确的。由于OP均未提及,因此我看不到这如何回答问题。
SiHa 2015年

3

对于给定的情况,此处可接受的答案是最正确的。

但是,这使我想知道一般只将布尔值取反。事实证明,这里公认的解决方案可以用作一种衬板,而另一种衬板也可以使用。假设您有一个已知为布尔值的变量“ n”,那么将其取反的最简单方法是:

n = n is False

这是我最初的解决方案,然后是这个问题的公认答案:

n = not n

后者更清晰,但是我想知道性能,并一直坚持下去timeit-事实证明,n = not n这也是反转布尔值的更快方法。


2

如果您尝试实现一个toggle,那么无论何时只要重新运行一个持久化的代码,它都会被否定,则可以实现以下目标:

try:
    toggle = not toggle
except NameError:
    toggle = True

运行此代码将首先将设置为toggleTrue并且每当此代码段调用时,切换都会被取消。

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.