Python - использовать метод [duplicate]

Цветной текст достаточно прост: прочитайте каждую строку и эхо его с соответствующими escape-последовательностями в начале и в конце. Но цветная стандартная ошибка становится сложной, потому что стандартная ошибка не передается в каналы.

Вот один из подходов, который работает путем замены стандартной ошибки и стандартного вывода, а затем фильтрации стандартного вывода.

Вот наша тестовая команда:

#!/bin/bash

echo hi
echo 'Error!' 1>&2

И скрипт wrapper:

#!/bin/bash
(# swap stderr and stdout
 exec 3>&1 # copy stdout to fd 3
 exec 1>&2 # copy stderr to fd 1
 exec 2>&3- # move saved stdout on fd 3 over to 2
 "${@}") | while read line; do
                echo -e "\033[31m${line}\033[0m"
           done

Затем:

$ ./wrapper ./test-command
hi
Error! # <- shows up red

К сожалению, весь вывод команды оболочки выходит из stderr, а не stdout, поэтому вы не можете передавать вывод в любые другие скрипты. Вероятно, вы можете обойти это с помощью создания временного fifo ... но, надеюсь, этого маленького сценария оболочки достаточно для удовлетворения ваших потребностей.

65
задан codeforester 28 September 2017 в 06:36
поделиться

11 ответов

Решение с использованием 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-one-one заменяется на xor-by-pre -put-constant-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__

131
ответ дан Raymond Hettinger 17 August 2018 в 11:47
поделиться
  • 1
    Последний пример кажется настолько гладким и интуитивным, но не работает в Python 3+ с удалением .next (). Есть ли способ заставить его работать аналогично в более поздней версии python? – labarna 18 July 2017 в 04:24
  • 2
    @labarna В Python 3, .next() заменена глобальной функцией next(). Вышеприведенный пример: toggle = itertools.cycle(...); next(toggle) – elpres 18 July 2017 в 04:50
  • 3
    toggle = itertools.cycle(['red', 'green', 'blue']) next(toggle) – Maximilian 18 July 2017 в 04:50
  • 4
    Пример XOR может быть обобщен для переключения между значениями a и b с помощью x = x ^ (a ^ b). – zxxc 18 July 2017 в 08:36

Вот еще один неинтуитивный способ. Красота - вы можете циклически перебирать несколько значений, а не только две [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
19
ответ дан Abhijit 17 August 2018 в 11:47
поделиться
  • 1
    да, это швайт как орех. спасибо всем, что это забавно смотреть на то, как разные люди подходят к проблеме (и информативны). – user 5 December 2011 в 08:51
  • 2
    Ницца, это миниатюрный автомат. – kindall 5 December 2011 в 09:30
  • 3
    ну, ваш самый интересный, но это не то, что мне лично нужно для того, о чем я спрашивал, так что хорошо, я думаю, что простая математика, вероятно, самая лучшая для меня, разве это не будет 1-х? – user 5 December 2011 в 17:02
  • 4
    Да, но это не должно меняться. – Blender 5 December 2011 в 19:02
  • 5
    ai, но это будет неправильно, хотя не так ли? некоторые отличные ответы здесь, SO скалы! – user 5 December 2011 в 22:02

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

toggle = not toggle

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

toggle = 'a' if toggle == 'b' else 'b'
16
ответ дан Blender 17 August 2018 в 11:47
поделиться
  • 1
    +1, но toggle = 0 if toggle else 1 короче и более общий – luc 5 December 2011 в 08:48
  • 2
    Извините, я поменяю переменные, чтобы сделать их более ясными. Я использовал inline if для переключения между двумя произвольными переменными, а не только 1 и 0. – Blender 5 December 2011 в 08:50
  • 3
    хорошо спасибо за разъяснение – luc 5 December 2011 в 08:55

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

def toggle(value):
    return 1 - value
1
ответ дан Bunny Rabbit 17 August 2018 в 11:47
поделиться

Тригонометрический подход, просто потому, что функции 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
4
ответ дан dani herrera 17 August 2018 в 11:47
поделиться

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

>>> 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
6
ответ дан hugo24 17 August 2018 в 11:47
поделиться

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

>>> 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)
4
ответ дан Ikem Krueger 17 August 2018 в 11:47
поделиться

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

1-x 

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

10
ответ дан M S 17 August 2018 в 11:47
поделиться
  • 1
    Поскольку (в любом случае Python 2.x) True и False являются целыми числами, хотя и с удивительно подробным методом __str__(), x также могут быть True или False здесь. Однако вы получите 1 или 0. – kindall 5 December 2011 в 09:32

Я использую функцию abs, очень полезную для циклов

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

x будет 0.

1
ответ дан Proteo5 17 August 2018 в 11:47
поделиться

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

p^=True

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

25
ответ дан renger 17 August 2018 в 11:47
поделиться
  • 1
    Отлично! p не нужно ссылаться дважды, чтобы этот метод работал! Идея, если вы переключите значение с длинной ссылкой. – ThorSummoner 12 December 2014 в 08:16
  • 2
    что называется этим оператором? – mix3d 20 September 2017 в 12:11
  • 3
    Это оператор XOR. – bastelflp 15 November 2017 в 00:44

Удивительно, что никто не упоминает хорошее старое разделение по модулю 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
7
ответ дан Yauhen Yakimovich 17 August 2018 в 11:47
поделиться
  • 1
    Я не уверен, как «pythonic» это (C-like) по модулю арифметика (i, e, применяется ли «pythonic»?). Я предполагаю, что это просто арифметика, работает везде, где у вас есть двоичный файл. – Yauhen Yakimovich 19 June 2013 в 13:02
  • 2
    Очевидно, машина конечного состояния с кортежем типа x = (1,2,3,0); token = 0; токен = x [токен] чрезвычайно интересен, поскольку он может быть даже более общим, чем просто групповая операция. – Yauhen Yakimovich 19 June 2013 в 13:08
Другие вопросы по тегам:

Похожие вопросы: