Цветной текст достаточно прост: прочитайте каждую строку и эхо его с соответствующими 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 ... но, надеюсь, этого маленького сценария оболочки достаточно для удовлетворения ваших потребностей.
Если значения являются логическими, самым быстрым подходом является использование оператора 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
Если значение переключается между 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.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__
Вот еще один неинтуитивный способ. Красота - вы можете циклически перебирать несколько значений, а не только две [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
Оператор not
отрицает вашу переменную (превращая ее в логическую, если она еще не одна). Вы можете , вероятно, использовать 1
и 0
взаимозаменяемо с True
и False
, поэтому просто отрицайте это:
toggle = not toggle
Но если вы используете два произвольных значения, используйте встроенный if
:
toggle = 'a' if toggle == 'b' else 'b'
if
для переключения между двумя произвольными i> переменными, а не только 1
и 0
.
– Blender
5 December 2011 в 08:50
Самый простой способ переключения между 1 и 0 состоит в вычитании из 1.
def toggle(value):
return 1 - value
Тригонометрический подход, просто потому, что функции 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
одним способом переключения является использование множественного присваивания
>>> 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
Использование обработчика исключений
>>> 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)
Только между 1 и 0, сделайте это
1-x
x может принимать 1 или 0
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.
Я всегда использую:
p^=True
Если p является логическим, оно переключается между истинным и ложным.
p
не нужно ссылаться дважды, чтобы этот метод работал! Идея, если вы переключите значение с длинной ссылкой.
– ThorSummoner
12 December 2014 в 08:16
Удивительно, что никто не упоминает хорошее старое разделение по модулю 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
.next()
заменена глобальной функциейnext()
. Вышеприведенный пример:toggle = itertools.cycle(...); next(toggle)
– elpres 18 July 2017 в 04:50toggle = itertools.cycle(['red', 'green', 'blue'])
next(toggle)
– Maximilian 18 July 2017 в 04:50a
иb
с помощьюx = x ^ (a ^ b)
. – zxxc 18 July 2017 в 08:36