Запустите команду оболочки с помощью сценария Python, дождитесь завершения и вернитесь к сценарию. - PullRequest
50 голосов
/ 28 ноября 2008

У меня есть скрипт на python, который должен запускать команду оболочки для каждого файла в каталоге:

import os

files = os.listdir(".")
for f in files:
    os.execlp("myscript", "myscript", f)

Это нормально работает для первого файла, но после завершения команды «myscript» выполнение останавливается и не возвращается к сценарию python.

Как я могу это сделать? Должен ли я fork() до calling os.execlp()?

Ответы [ 7 ]

86 голосов
/ 28 ноября 2008

подпроцесс: модуль subprocess позволяет создавать новые процессы, подключиться к их вводу / выводу / ошибке трубы, и получите их коды возврата.

http://docs.python.org/library/subprocess.html

Использование:

import subprocess
process = subprocess.Popen(command, shell=True, stdout=subprocess.PIPE)
process.wait()
print process.returncode
59 голосов
/ 28 ноября 2008

Вы можете использовать subprocess.Popen. Есть несколько способов сделать это:

import subprocess
cmd = ['/run/myscript', '--arg', 'value']
p = subprocess.Popen(cmd, stdout=subprocess.PIPE)
for line in p.stdout:
    print line
p.wait()
print p.returncode

Или, если вам все равно, что на самом деле делает внешняя программа:

cmd = ['/run/myscript', '--arg', 'value']
subprocess.Popen(cmd).wait()
12 голосов
/ 03 марта 2011

Модуль подпроцесса появился на свет с 2008 года. В частности check_call и check_output делают простые подпроцессы еще проще. Семейство функций check_* приятно, что они вызывают исключение, если что-то идет не так.

import os
import subprocess

files = os.listdir('.')
for f in files:
   subprocess.check_call( [ 'myscript', f ] )

Любой вывод, сгенерированный myscript, будет отображаться так, как будто ваш процесс произвел вывод (технически myscript и ваш скрипт на python используют один и тот же стандартный вывод). Есть несколько способов избежать этого.

  • check_call( [ 'myscript', f ], stdout=subprocess.PIPE )
    Stdout будет подавлен (будьте осторожны, если myscript производит более 4 тыс. Выводов). stderr все равно будет отображаться, если вы не добавите опцию stderr=subprocess.PIPE.
  • check_output( [ 'myscript', f ] )
    check_output возвращает стандартный вывод в виде строки, поэтому он не отображается. stderr по-прежнему отображается, если вы не добавите параметр stderr=subprocess.STDOUT.
7 голосов
/ 28 ноября 2008

Функции os.exec*() заменяют текущей программы на новую. Когда эта программа заканчивается, ваш процесс завершается. Вы, вероятно, хотите os.system().

3 голосов
/ 28 ноября 2008

использовать Spawn

import os
os.spawnlp(os.P_WAIT, 'cp', 'cp', 'index.html', '/dev/null')
0 голосов
/ 03 декабря 2018

это сработало для меня нормально!

shell_command = "ls -l" subprocess.call(shell_command.split())

0 голосов
/ 08 марта 2018

Я использую os.system

import os
os.system("pdftoppm -png {} {}".format(path2pdf, os.path.join(tmpdirname, "temp")))
...