Quelle est la manière la plus efficace de basculer entre 0 et 1?

121
user1064306 5 déc. 2011 à 10:30

16 réponses

Meilleure réponse

Solution utilisant NOT

Si les valeurs sont booléennes, l'approche la plus rapide consiste à utiliser not :

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

Solution utilisant la soustraction

Si les valeurs sont numériques, la soustraction du total est un moyen simple et rapide de basculer les valeurs:

>>> 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

Solution utilisant XOR

Si la valeur bascule entre 0 et 1 , vous pouvez utiliser un bitwise exclusif-ou:

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

La technique se généralise à n'importe quelle paire d'entiers. L'étape xor-by-one est remplacée par une constante xor-by-precomputed:

>>> 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

(Cette idée a été soumise par Nick Coghlan et plus tard généralisée par @zxxc.)

Solution utilisant un dictionnaire

Si les valeurs sont hachables, vous pouvez utiliser un dictionnaire:

>>> 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'

Solution utilisant une expression conditionnelle

Le moyen le plus lent consiste à utiliser une expression conditionnelle:

>>> 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]

Solution utilisant itertools

Si vous avez plus de deux valeurs, itertools.cycle () La fonction fournit un moyen générique rapide de basculer entre les valeurs successives:

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

Notez que dans Python 3, la méthode next() a été changée en __next__(), donc la première ligne serait maintenant écrite comme toggle = itertools.cycle(['red', 'green', 'blue']).__next__

256
s3cur3 31 janv. 2019 à 23:44

Voici une autre manière non intuitive. La beauté est que vous pouvez parcourir plusieurs valeurs et pas seulement deux [0,1]

Pour deux valeurs (basculement)

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

Pour plusieurs valeurs (disons 4)

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

Je ne m'attendais pas à ce que cette solution soit aussi la plus rapide

>>> 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
22
Abhijit 5 déc. 2011 à 07:43

J'utilise toujours:

p^=True

Si p est un booléen, cela bascule entre vrai et faux.

31
renger 10 mai 2014 à 14:58

Une façon de basculer est d'utiliser l'attribution multiple

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

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

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

Utilisation d'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
6
hugo24 19 juin 2013 à 14:20

La commutation entre -1 et +1 peut être obtenue par multiplication en ligne; utilisé pour le calcul de pi de la manière 'Leibniz' (ou similaire):

sign = 1
result = 0
for i in range(100000):
    result += 1 / (2*i + 1) * sign
    sign *= -1
print("pi (estimate): ", result*4)
0
John Bograd 6 févr. 2019 à 17:07

Les variables a et b peuvent être TOUTES deux valeurs, comme 0 et 1, ou 117 et 711, ou "têtes" et "queues". Aucun calcul mathématique n'est utilisé, juste un échange rapide des valeurs chaque fois qu'une bascule est souhaitée.

a = True   
b = False   

a,b = b,a   # a is now False
a,b = b,a   # a is now True
2
user2948775 6 févr. 2019 à 00:11

Approche trigonométrique , simplement parce que les fonctions sin et cos sont cool.

enter image description here

>>> 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
11
dani herrera 18 juil. 2017 à 10:45

Utilisation du gestionnaire d'exceptions

>>> 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

Ok, je suis le pire:

enter image description here

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)
4
Ikem Krueger 23 févr. 2018 à 04:27

Étonnamment, personne ne mentionne le bon vieux module de division 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

Notez qu'il est équivalent à x = x - 1, mais l'avantage de la technique modulo est que la taille du groupe ou la longueur de l'intervalle peut être supérieure à seulement 2 éléments, vous donnant ainsi un schéma d'entrelacement à tour de rôle similaire à boucle.

Maintenant, juste pour 2, le basculement peut être un peu plus court (en utilisant l'opérateur au niveau du bit):

x = x ^ 1
7
Yauhen Yakimovich 19 juin 2013 à 12:04

Que diriez-vous d'une bascule imaginaire qui stocke non seulement la bascule actuelle, mais quelques autres valeurs qui lui sont associées?

toggle = complex.conjugate

Stockez toute valeur + ou - à gauche et toute valeur non signée à droite:

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

Zero fonctionne aussi:

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

Récupérez facilement la valeur de bascule actuelle (True et False représentent + et -), la valeur LHS (réelle) ou RHS (imaginaire):

>>> 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

Échangez facilement LHS et RHS (mais notez que le signe des deux valeurs ne doit pas être important):

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

Permutez facilement LHS et RHS et aussi en même temps:

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

Protège contre les erreurs:

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

Effectuer des modifications sur LHS et RHS:

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

... mais attention à manipuler le RHS:

>>> z = 1-1j
>>> z += 2j
>>> z
(1+1j) # whoops! toggled it!
3
Rick supports Monica 1 févr. 2019 à 14:58

Si vous avez affaire à une variable entière, vous pouvez incrémenter 1 et limiter votre ensemble à 0 et 1 (mod)

X = 0  # or X = 1
X = (X + 1)%2
0
Italo Nesi 5 févr. 2019 à 22:04

L'opérateur not annule votre variable (la convertissant en booléen si ce n'est déjà fait). Vous pouvez probablement utiliser 1 et 0 de manière interchangeable avec True et False, alors niez-le simplement:

toggle = not toggle

Mais si vous utilisez deux valeurs arbitraires, utilisez un if en ligne:

toggle = 'a' if toggle == 'b' else 'b'
18
Blender 21 juin 2013 à 18:13

Faisons un peu de piratage de trame. Basculez une variable par son nom. Remarque: cela peut ne pas fonctionner avec chaque runtime Python.

Disons que vous avez une variable "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
Brantley Harris 1 févr. 2019 à 19:53

J'utilise la fonction abs, très utile sur les boucles

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

X sera 0.

1
Proteo5 24 oct. 2017 à 18:17

Entre 1 et 0, faites ceci

1-x 

X peut prendre 1 ou 0

14
M S 5 déc. 2011 à 06:33

Le moyen le plus simple de basculer entre 1 et 0 est de soustraire de 1.

def toggle(value):
    return 1 - value
4
Bunny Rabbit 4 août 2017 à 06:46