Как распечатать на stderr в Python?
Я встречал несколько способов написать в stderr:
# Note: this first one does not work in Python 3
print >> sys.stderr, "spam"
sys.stderr.write("spam\n")
os.write(2, b"spam\n")
from __future__ import print_function
print("spam", file=sys.stderr)
Кажется, что это противоречит дзен Python # 13 †, так каков предпочтительный способ сделать это? Есть ли какие-либо преимущества или недостатки в том или ином виде?
† Должен быть один - и желательно только один - очевидный способ сделать это.
19 ответов
Я обнаружил, что это единственный короткий + гибкий + портативный + читаемый:
from __future__ import print_function
import sys
def eprint(*args, **kwargs):
print(*args, file=sys.stderr, **kwargs)
Функция eprint
может использоваться так же, как стандарт print
функция:
>>> print("Test")
Test
>>> eprint("Test")
Test
>>> eprint("foo", "bar", "baz", sep="---")
foo---bar---baz
import sys
sys.stderr.write()
Это мой выбор, просто более читаемый и точно говорящий о том, что вы собираетесь делать, и переносимый между версиями.
Изменить: "pythonic" - это третья мысль о читабельности и производительности... с учетом этих двух вещей, с python 80% вашего кода будет pythonic. понимание списка - это "большая вещь", которая используется не так часто (удобочитаемость).
print >> sys.stderr
ушел в Python3. http://docs.python.org/3.0/whatsnew/3.0.html говорит:
Old: print >>sys.stderr, "fatal error"
New: print("fatal error", file=sys.stderr)
К сожалению, это довольно некрасиво. В качестве альтернативы используйте
sys.stderr.write("fatal error\n")
но обратите внимание, что write
не является заменой 1:1 для print
,
Никто не упоминается logging
пока, но регистрация была создана специально для сообщения об ошибках. По умолчанию он настроен на запись в stderr. Этот скрипт:
# foo.py
import logging
logging.basicConfig(format='%(message)s')
logging.warning('I print to stderr by default')
logging.info('For this you must change the level and add a handler.')
print('hello world')
имеет следующий результат при запуске в командной строке:
$ python3 foo.py > bar.txt
I print to stderr by default
(и bar.txt содержит "привет мир")
(Заметка, logging.warn
устарел, используйте logging.warning
вместо)
Для Python 2 мой выбор:print >> sys.stderr, 'spam'
Потому что вы можете просто напечатать списки / диктанты и т. Д. Без преобразования в строку. print >> sys.stderr, {'spam': 'spam'}
вместо:sys.stderr.write(str({'spam': 'spam'}))
Я бы сказал, что ваш первый подход:
print >> sys.stderr, 'spam'
это "Один. . . очевидный способ сделать это" Другие не удовлетворяют правилу № 1 ("Красиво лучше, чем безобразно".)
Я сделал следующее, используя Python 3:
from sys import stderr
def print_err(*args, **kwargs):
print(*args, file=stderr, **kwargs)
Так что теперь я могу добавить ключевые аргументы, например, чтобы избежать возврата каретки:
print_err("Error: end of the file reached. The word ", end='')
print_err(word, "was not found")
В Python 3 можно просто использовать print ():
print(*objects, sep=' ', end='\n', file=sys.stdout, flush=False)
почти из коробки:
import sys
print("Hello, world!", file=sys.stderr)
или же:
from sys import stderr
print("Hello, world!", file=stderr)
Это просто и не должно включать ничего кроме sys.stderr
,
Это будет имитировать стандартную функцию печати, но вывод на stderr
def print_err(*args):
sys.stderr.write(' '.join(map(str,args)) + '\n')
РЕДАКТИРОВАТЬ В ретроспективе, я думаю, что потенциальная путаница с изменением sys.stderr и отсутствием обновленного поведения делает этот ответ не таким хорошим, как просто использование простой функции, как указывали другие.
Использование только частичного экономит вам 1 строку кода. Потенциальная путаница не стоит сохранять 1 строку кода.
оригинал
Чтобы сделать это еще проще, вот версия, в которой используется "частичный", что очень помогает при переносе функций.
from __future__ import print_function
import sys
from functools import partial
error = partial(print, file=sys.stderr)
Вы тогда используете это так
error('An error occured!')
Вы можете проверить, что он печатает в stderr, а не в stdout, выполнив следующие действия (переопределение кода с http://coreygoldberg.blogspot.com.au/2009/05/python-redirect-or-turn-off-stdout-and.html):
# over-ride stderr to prove that this function works.
class NullDevice():
def write(self, s):
pass
sys.stderr = NullDevice()
# we must import print error AFTER we've removed the null device because
# it has been assigned and will not be re-evaluated.
# assume error function is in print_error.py
from print_error import error
# no message should be printed
error("You won't see this error!")
Недостатком этого является частичное назначение значения sys.stderr функции-оболочке во время создания. Это означает, что если вы перенаправите stderr позже, это не повлияет на эту функцию. Если вы планируете перенаправить stderr, используйте метод **kwargs, упомянутый aaguirre на этой странице.
То же самое относится к stdout:
print 'spam'
sys.stdout.write('spam\n')
Как указывалось в других ответах, print предлагает симпатичный интерфейс, который часто более удобен (например, для печати отладочной информации), в то время как запись выполняется быстрее и также может быть более удобной, когда необходимо точно отформатировать вывод. Я бы также подумал о ремонтопригодности:
Позже вы можете переключиться между stdout/stderr и обычным файлом.
Синтаксисprint() изменился в Python 3, поэтому, если вам нужно поддерживать обе версии, лучше использовать write().
Я работаю в Python 3.4.3. Я вырезал небольшой набор текста, который показывает, как я попал сюда:
[18:19 jsilverman@JSILVERMAN-LT7 pexpect]$ python3
>>> import sys
>>> print("testing", file=sys.stderr)
testing
>>>
[18:19 jsilverman@JSILVERMAN-LT7 pexpect]$
Это сработало? Попробуйте перенаправить stderr в файл и посмотрите, что произойдет:
[18:22 jsilverman@JSILVERMAN-LT7 pexpect]$ python3 2> /tmp/test.txt
>>> import sys
>>> print("testing", file=sys.stderr)
>>> [18:22 jsilverman@JSILVERMAN-LT7 pexpect]$
[18:22 jsilverman@JSILVERMAN-LT7 pexpect]$ cat /tmp/test.txt
Python 3.4.3 (default, May 5 2015, 17:58:45)
[GCC 4.9.2] on cygwin
Type "help", "copyright", "credits" or "license" for more information.
testing
[18:22 jsilverman@JSILVERMAN-LT7 pexpect]$
Ну, кроме того факта, что небольшое введение, которое дает вам Python, было добавлено в stderr (куда еще он пойдет?), Это работает.
Если вы хотите выйти из программы из-за фатальной ошибки, используйте:
sys.exit("Your program caused a fatal error. ... description ...")
а также import sys
в шапке.
Если вы делаете простой тест:
import time
import sys
def run1(runs):
x = 0
cur = time.time()
while x < runs:
x += 1
print >> sys.stderr, 'X'
elapsed = (time.time()-cur)
return elapsed
def run2(runs):
x = 0
cur = time.time()
while x < runs:
x += 1
sys.stderr.write('X\n')
sys.stderr.flush()
elapsed = (time.time()-cur)
return elapsed
def compare(runs):
sum1, sum2 = 0, 0
x = 0
while x < runs:
x += 1
sum1 += run1(runs)
sum2 += run2(runs)
return sum1, sum2
if __name__ == '__main__':
s1, s2 = compare(1000)
print "Using (print >> sys.stderr, 'X'): %s" %(s1)
print "Using (sys.stderr.write('X'),sys.stderr.flush()):%s" %(s2)
print "Ratio: %f" %(float(s1) / float(s2))
Вы обнаружите, что sys.stderr.write() последовательно в 1,81 раза быстрее!
import logging
logging.basicConfig(format='[%(levelname)s] %(message)s')
logging.error('is error, alarm!')
logging.warning('is simple waring')
print('hello')
регистрация pydoc
Я делаю это просто для удовольствия, но вот еще один способ... :-)
message = 'error: Belly up!!'
print(message, file=sys.stderr if 'error' in message.lower() else sys.stdout)
Ответ на вопрос: есть разные способы напечатать stderr в python, но это зависит от 1.) какую версию python мы используем 2.) какой именно вывод мы хотим.
Различие между print и функцией записи stderr: stderr: stderr (стандартная ошибка) - это канал, который встроен в каждую систему UNIX/Linux, когда ваша программа аварийно завершает работу и выводит отладочную информацию (например, трассировку в Python), она переходит к stderr труба.
print: print - это оболочка, которая форматирует входные данные (входные данные - это пробел между аргументом и символом новой строки в конце), а затем вызывает функцию записи для данного объекта, по умолчанию это sys.stdout, но мы можем передать файл, т. е. мы также можем напечатать вход в файл.
Python2: если мы используем python2, то
>>> import sys
>>> print "hi"
hi
>>> print("hi")
hi
>>> print >> sys.stderr.write("hi")
hi
Завершающая запятая в Python2 в Python3 стала параметром, поэтому если мы используем запятые, чтобы избежать перехода на новую строку после печати, в Python3 это будет выглядеть как print('Text to print', end=' '), что является синтаксической ошибкой в Python2,
http://python3porting.com/noconv.html
Если мы проверим тот же сценарий выше в python3:
>>> import sys
>>> print("hi")
hi
В Python2.6 есть возможность импорта в функцию в будущем. Поэтому, чтобы избежать любых синтаксических ошибок и других различий, мы должны начать любой файл, в котором мы используем print(), из будущего импорта print_function. Будущий импорт работает только в Python2.6 и более поздних версиях, поэтому для Python2.5 и более ранних версий у вас есть два варианта. Вы можете преобразовать более сложную печать во что-то более простое или использовать отдельную функцию печати, которая работает как в Python2, так и в Python3.
>>> from __future__ import print_function
>>>
>>> def printex(*args, **kwargs):
... print(*args, file=sys.stderr, **kwargs)
...
>>> printex("hii")
hii
>>>
Случай: Следует отметить, что sys.stderr.write() или sys.stdout.write() ( stdout (стандартный вывод) - это канал, который встроен в каждую систему UNIX/Linux) не является заменой для печати, но да мы можем использовать его как альтернативу в некоторых случаях. Print - это оболочка, которая оборачивает ввод пробелом и символом новой строки в конце и использует функцию записи для записи. По этой причине sys.stderr.write() работает быстрее.
Примечание: мы также можем отслеживать и отлаживать, используя ведение журнала
#test.py
import logging
logging.info('This is the existing protocol.')
FORMAT = "%(asctime)-15s %(clientip)s %(user)-8s %(message)s"
logging.basicConfig(format=FORMAT)
d = {'clientip': '192.168.0.1', 'user': 'fbloggs'}
logging.warning("Protocol problem: %s", "connection reset", extra=d)
Другой путь
import sys
print("{}".format(sys.exec_info()[1], file=sys.stderr)