Как переключить значение в Python

128

Каков наиболее эффективный способ переключения между 0и 1?

codeforester
источник
в то время как в этом вопросе спрашивается, как наиболее эффективно переключать значения двоичным способом, некоторые ответы объясняют циклическое переключение (произвольных) значений, например stackoverflow.com/a/61041907/537865
mad,

Ответы:

273

Решение с использованием NOT

Если значения являются логическими, самым быстрым подходом является использование оператора not :

>>> x = True
>>> x = not x        # toggle
>>> x
False
>>> x = not x        # toggle
>>> x
True
>>> x = not x        # toggle
>>> x
False

Решение с использованием вычитания

Если значения числовые, то вычитание из суммы - это простой и быстрый способ переключения значений:

>>> A = 5
>>> B = 3
>>> total = A + B
>>> x = A
>>> x = total - x    # toggle
>>> x
3
>>> x = total - x    # toggle
>>> x
5
>>> x = total - x    # toggle
>>> x
3

Решение с использованием XOR

Если значение переключается между 0 и 1 , вы можете использовать побитовое исключающее ИЛИ :

>>> x = 1
>>> x ^= 1
>>> x
0
>>> x ^= 1
>>> x
1

Техника распространяется на любую пару целых чисел. Шаг xor-by-one заменяется на xor-by-precomputed-constant:

>>> A = 205
>>> B = -117
>>> t = A ^ B        # precomputed toggle constant
>>> x = A
>>> x ^= t           # toggle
>>> x
-117
>>> x ^= t           # toggle
>>> x
205
>>> x ^= t           # toggle
>>> x
-117

(Эта идея была предложена Ником Когланом, а затем обобщена @zxxc.)

Решение с использованием словаря

Если значения хешируются, вы можете использовать словарь:

>>> A = 'xyz'
>>> B = 'pdq'
>>> d = {A:B, B:A}
>>> x = A
>>> x = d[x]         # toggle
>>> x
'pdq'
>>> x = d[x]         # toggle
>>> x
'xyz'
>>> x = d[x]         # toggle
>>> x
'pdq'

Решение с использованием условного выражения

Самый медленный способ - использовать условное выражение :

>>> A = [1,2,3]
>>> B = [4,5,6]
>>> x = A
>>> x = B if x == A else A
>>> x
[4, 5, 6]
>>> x = B if x == A else A
>>> x
[1, 2, 3]
>>> x = B if x == A else A
>>> x
[4, 5, 6]

Решение с использованием itertools

Если у вас более двух значений, функция itertools.cycle () предоставляет общий быстрый способ переключения между последовательными значениями:

>>> import itertools
>>> toggle = itertools.cycle(['red', 'green', 'blue']).next
>>> toggle()
'red'
>>> toggle()
'green'
>>> toggle()
'blue'
>>> toggle()
'red'
>>> toggle()
'green'
>>> toggle()
'blue'

Обратите внимание, что в Python 3 next()метод был изменен на __next__(), поэтому первая строка теперь будет записана какtoggle = itertools.cycle(['red', 'green', 'blue']).__next__

Раймонд Хеттингер
источник
Последний пример кажется таким гладким и интуитивно понятным, но не работает в Python 3+ с удалением .next (). Есть ли способ заставить его работать аналогичным образом в более поздней версии Python?
labarna
2
@labarna В Python 3, .next()был заменен глобальной next()функцией. Приведенный выше пример будет выглядеть так:toggle = itertools.cycle(...); next(toggle)
elpres
2
toggle = itertools.cycle(['red', 'green', 'blue']) next(toggle)
Максимилиан
7
Пример XOR можно обобщить для переключения между значениями aи bиспользованием x = x ^ (a ^ b).
zxxc
int(not 0)и int(not 1)... хммм
jhrr
33

Я всегда использую:

p^=True

Если p является логическим, это переключается между истиной и ложью.

Ренгер
источник
1
Отлично! pне нужно ссылаться дважды, чтобы этот метод работал !! Идея, если вы переключаете значение с длинной длинной ссылкой.
ThorSummoner
1
как называется этот оператор?
mix3d
4
Это оператор XOR.
bastelflp
1
@ mix3d Точнее, это «побитовое исключающее ИЛИ» (в отличие от «логическое исключающее ИЛИ ») - wiki.python.org/moin/BitwiseOperators . Логический XOR не имеет специального оператора в Python в целом, но вы можете найти его реализованным в некоторых особых случаях, например, в модуле decimal.
Тейлор
@ mix3d ^=- побитовое присвоение xor
wjandrea 06
23

Вот еще один не интуитивный способ. Прелесть в том, что вы можете перебирать несколько значений, а не только два [0,1]

Для двух значений (переключение)

>>> x=[1,0]
>>> toggle=x[toggle]

Для нескольких значений (скажем, 4)

>>> x=[1,2,3,0]
>>> toggle=x[toggle]

Я тоже не ожидал, что это решение будет чуть ли не самым быстрым

>>> stmt1="""
toggle=0
for i in xrange(0,100):
    toggle = 1 if toggle == 0 else 0
"""
>>> stmt2="""
x=[1,0]
toggle=0
for i in xrange(0,100):
    toggle=x[toggle]
"""
>>> t1=timeit.Timer(stmt=stmt1)
>>> t2=timeit.Timer(stmt=stmt2)
>>> print "%.2f usec/pass" % (1000000 * t1.timeit(number=100000)/100000)
7.07 usec/pass
>>> print "%.2f usec/pass" % (1000000 * t2.timeit(number=100000)/100000)
6.19 usec/pass
stmt3="""
toggle = False
for i in xrange(0,100):
    toggle = (not toggle) & 1
"""
>>> t3=timeit.Timer(stmt=stmt3)
>>> print "%.2f usec/pass" % (1000000 * t3.timeit(number=100000)/100000)
9.84 usec/pass
>>> stmt4="""
x=0
for i in xrange(0,100):
    x=x-1
"""
>>> t4=timeit.Timer(stmt=stmt4)
>>> print "%.2f usec/pass" % (1000000 * t4.timeit(number=100000)/100000)
6.32 usec/pass
Abhijit
источник
1
да это швит как орех. спасибо всем, интересно посмотреть, как разные люди подходят к проблеме (и информативно).
Красиво, это миниатюрная государственная машина.
kindall
ну, ваш самый интересный, но это не то, что мне лично нужно для того, о чем я спрашивал, так что хорошо, я думаю, что тогда простая математика, вероятно, лучше для меня, разве это не будет 1-x?
Да, но от этого скорость не изменится.
Blender
ай, но это сделало бы это неправильно, хотя не так ли? некоторые отличные ответы здесь, ТАК классно!
19

notОператор отрицает переменную (превращая его в логическое значение , если оно уже не один). Вероятно, вы можете использовать 1и 0взаимозаменяемо с Trueи False, поэтому просто отрицайте его:

toggle = not toggle

Но если вы используете два произвольных значения, используйте встроенный if:

toggle = 'a' if toggle == 'b' else 'b'
смеситель
источник
1
+1, но toggle = 0 if toggle else 1короче и более общий
люк
Извините, я поменяю переменные, чтобы было понятнее. Я использовал встроенный ifдля переключения между двумя произвольными переменными, а не только 1и 0.
Blender
14

Только между 1 и 0, сделайте это

1-x 

x может принимать 1 или 0

РС
источник
Поскольку (во всяком случае в Python 2.x) Trueи Falseна самом деле являются целыми числами, хотя и с удивительно подробным __str__()методом, xтакже могут быть Trueили Falseздесь. Тем не менее, вы получите 1 или 0 обратно.
kindall 05
12

Тригонометрический подход , просто потому что sinи cosфункции классные.

введите описание изображения здесь

>>> import math
>>> def generator01():
...     n=0
...     while True:
...         yield abs( int( math.cos( n * 0.5 * math.pi  ) ) )
...         n+=1
... 
>>> g=generator01() 
>>> g.next()
1
>>> g.next()
0
>>> g.next()
1
>>> g.next()
0
Дани Эррера
источник
О Боже! Я <3 тебя.
Ришаб Аграхари
2
@RishabhAgrahari, да, чувак, я победил в испытании Раймонда Хеттингера ;)
дани Эррера
7

Удивительно, но никто не упомянул старое доброе деление по модулю 2:

In : x = (x + 1)  % 2 ; x
Out: 1

In : x = (x + 1)  % 2 ; x
Out: 0

In : x = (x + 1)  % 2 ; x
Out: 1

In : x = (x + 1)  % 2 ; x
Out: 0

Обратите внимание, что это эквивалентно x = x - 1, но преимущество метода по модулю состоит в том, что размер группы или длина интервала могут быть больше, чем всего 2 элемента, что дает вам аналогичную циклической схеме чередования.

Теперь только для 2 переключение может быть немного короче (с использованием побитового оператора):

x = x ^ 1
Евгений Якимович
источник
Я не уверен, насколько «питонический» это (C-подобный) по модулю арифметики (то есть, применяется ли «питонический»?). Я думаю, это просто арифметика, работает везде, где есть двоичный файл.
Евгений Якимович
Очевидно, конечный автомат с кортежем вида x = (1,2,3,0); token = 0; token = x [token] чрезвычайно интересен, поскольку может быть даже более общим, чем просто групповая операция.
Евгений Якимович
7

один из способов переключения - использование множественного назначения

>>> a = 5
>>> b = 3

>>> t = a, b = b, a
>>> t[0]
3

>>> t = a, b = b, a
>>> t[0]
5

Используя itertools:

In [12]: foo = itertools.cycle([1, 2, 3])

In [13]: next(foo)
Out[13]: 1

In [14]: next(foo)
Out[14]: 2

In [15]: next(foo)
Out[15]: 3

In [16]: next(foo)
Out[16]: 1

In [17]: next(foo)
Out[17]: 2
hugo24
источник
4

Самый простой способ переключаться между 1 и 0 - вычесть из 1.

def toggle(value):
    return 1 - value
Кролик кролик
источник
4

Использование обработчика исключений

>>> def toogle(x):
...     try:
...         return x/x-x/x
...     except  ZeroDivisionError:
...         return 1
... 
>>> x=0
>>> x=toogle(x)
>>> x
1
>>> x=toogle(x)
>>> x
0
>>> x=toogle(x)
>>> x
1
>>> x=toogle(x)
>>> x
0

Хорошо, я худший:

введите описание изображения здесь

import math
import sys

d={1:0,0:1}
l=[1,0]

def exception_approach(x):
    try:
        return x/x-x/x
    except  ZeroDivisionError:
        return 1

def cosinus_approach(x):
    return abs( int( math.cos( x * 0.5 * math.pi  ) ) )

def module_approach(x):
    return  (x + 1)  % 2

def subs_approach(x):
    return  x - 1

def if_approach(x):
    return 0 if x == 1 else 1

def list_approach(x):
    global l
    return l[x]

def dict_approach(x):
    global d
    return d[x]

def xor_approach(x):
    return x^1

def not_approach(x):
    b=bool(x)
    p=not b
    return int(p)

funcs=[ exception_approach, cosinus_approach, dict_approach, module_approach, subs_approach, if_approach, list_approach, xor_approach, not_approach ]

f=funcs[int(sys.argv[1])]
print "\n\n\n", f.func_name
x=0
for _ in range(0,100000000):
    x=f(x)
Дани Эррера
источник
3

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

toggle = complex.conjugate

Сохраните любое значение + или - слева и любое значение без знака справа:

>>> x = 2 - 3j
>>> toggle(x)
(2+3j)

Ноль тоже работает:

>>> y = -2 - 0j
>>> toggle(y)
(-2+0j)

Легко получить текущее значение переключения ( Trueи Falseпредставить + и -), LHS (реальное) значение или RHS (мнимое) значение:

>>> import math
>>> curr = lambda i: math.atan2(i.imag, -abs(i.imag)) > 0
>>> lhs = lambda i: i.real
>>> rhs = lambda i: abs(i.imag)
>>> x = toggle(x)
>>> curr(x)
True
>>> lhs(x)
2.0
>>> rhs(x)
3.0

Легко поменять местами LHS и RHS (но учтите, что знак обоих значений не должен иметь значения):

>>> swap = lambda i: i/-1j
>>> swap(2+0j)
2j
>>> swap(3+2j)
(2+3j)

Легко меняйте местами LHS и RHS, а также переключайтесь одновременно:

>>> swaggle = lambda i: i/1j
>>> swaggle(2+0j)
-2j
>>> swaggle(3+2j)
(2-3j)

Защита от ошибок:

>>> toggle(1)
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
TypeError: descriptor 'conjugate' requires a 'complex' object but received a 'int'

Внесите изменения в LHS и RHS:

>>> x += 1+2j
>>> x
(3+5j)

... но будьте осторожны, манипулируя RHS:

>>> z = 1-1j
>>> z += 2j
>>> z
(1+1j) # whoops! toggled it!
Рик поддерживает Монику
источник
2

Переменные a и b могут иметь ЛЮБЫЕ два значения, например, 0 и 1, или 117 и 711, или «орел» и «решка». Никаких математических вычислений не используется, только быстрая замена значений каждый раз, когда требуется переключение.

a = True   
b = False   

a,b = b,a   # a is now False
a,b = b,a   # a is now True
user2948775
источник
1

Я использую функцию абс, очень полезную в циклах

x = 1
for y in range(0, 3):
    x = abs(x - 1)

x будет 0.

Proteo5
источник
0

Давайте сделаем взлом кадра. Переключить переменную по имени. Примечание. Это может работать не во всех средах выполнения Python.

Скажем, у вас есть переменная "x"

>>> import inspect
>>> def toggle(var_name):
>>>     frame = inspect.currentframe().f_back
>>>     vars = frame.f_locals
>>>     vars[var_name] = 0 if vars[var_name] == 1 else 1

>>> x = 0
>>> toggle('x')
>>> x
1
>>> toggle('x')
>>> x
0
Брантли Харрис
источник
0

Если вы имеете дело с целочисленной переменной, вы можете увеличить на 1 и ограничить свой набор значениями 0 и 1 (мод)

X = 0  # or X = 1
X = (X + 1)%2
Итало Неси
источник
0

Переключение между -1 и +1 может быть получено встроенным умножением; используется для вычисления числа Пи способом Лейбница (или аналогичным):

sign = 1
result = 0
for i in range(100000):
    result += 1 / (2*i + 1) * sign
    sign *= -1
print("pi (estimate): ", result*4)
Джон Боград
источник
0

Вы можете использовать из indexв listс.

def toggleValues(values, currentValue):
    return values[(values.index(currentValue) + 1) % len(values)]

> toggleValues( [0,1] , 1 )
> 0
> toggleValues( ["one","two","three"] , "one" )
> "two"
> toggleValues( ["one","two","three"] , "three")
> "one"

Плюсы : Отсутствие дополнительных библиотек, самостоятельный поясняющий код и работа с произвольными типами данных.

Минусы : нет дубликата-сохранения. toggleValues(["one","two","duped", "three", "duped", "four"], "duped") всегда вернусь"three"

Сумасшедший
источник