Как сделать булевую алгебру по отсутствующим значениям?

Я хочу реплицировать значения boolean NA, поскольку они ведут себя в R:

NA является допустимым логическим объектом. Если компонент x или y является NA, результатом будет NA, если результат неоднозначен. Другими словами, NA и TRUE оценивают NA, но NA и FALSE оцениваются как FALSE. http://stat.ethz.ch/R-manual/R-devel/library/base/html/Logic.html

Я видел, что None рекомендуется для отсутствующих значений, но Python преобразует None в False при оценке булевых выражений и вычисляет от None or False до False. Разумеется, результат должен был None, так как не было сделано никаких выводов, учитывая недостающее значение.

Как добиться этого в Python?

EDIT Принятый ответ правильно вычисляется с помощью побитовых булевых операторов, но для достижения такого же поведения с логическими операторами not, or и and, кажется, требуется изменение в программировании на Python язык.

Ответы

Ответ 1

Как говорили другие, вы можете определить свой собственный класс.

class NA_(object):
    instance = None # Singleton (so `val is NA` will work)
    def __new__(self):
        if NA_.instance is None:
            NA_.instance = super(NA_, self).__new__(self)
        return NA_.instance
    def __str__(self): return "NA"
    def __repr__(self): return "NA_()"
    def __and__(self, other):
        if self is other or other:
            return self
        else:
            return other
    __rand__ = __and__
    def __or__(self, other):
        if self is other or other:
            return other
        else:
            return self
    __ror__ = __or__
    def __xor__(self, other):
        return self
    __rxor__ = __xor__
    def __eq__(self, other):
        return self is other
    __req__ = __eq__
    def __nonzero__(self):
        raise TypeError("bool(NA) is undefined.")
NA = NA_()

Использование:

>>> print NA & NA
NA
>>> print NA & True
NA
>>> print NA & False
False
>>> print NA | True
True
>>> print NA | False
NA
>>> print NA | NA
NA
>>> print NA ^ True
NA
>>> print NA ^ NA
NA
>>> if NA: print 3
...
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
  File "<stdin>", line 28, in __nonzero__
TypeError: bool(NA) is undefined.
>>> if NA & False: print 3
...
>>>
>>> if NA | True: print 3
...
3
>>>

Ответ 2

Вы можете сделать это, создав класс и переопределив логические методы работы.

>>> class NA_type(object):
        def __and__(self,other):
                if other == True:
                        return self
                else:
                        return False
        def __str__(self):
                return 'NA'


>>> 
>>> NA = NA_type()
>>> print NA & True
NA
>>> print NA & False
False