我有这样的命令。
wmctrl -lp | awk '/gedit/ { print $1 }'
我希望它的输出在python脚本中,我尝试了这段代码
>>> import subprocess
>>> proc = subprocess.Popen(["wmctrl -lp", "|","awk '/gedit/ {print $1}"], shell=True, stdout=subprocess.PIPE, stderr=subprocess.PIPE)
>>> proc.stdout.readline()
'0x0160001b -1 6504 beer-laptop x-nautilus-desktop\n'
>>> proc.stdout.readline()
'0x0352f117 0 6963 beer-laptop How to get output from external command combine with Pipe - Stack Overflow - Chromium\n'
>>> proc.stdout.readline()
'0x01400003 -1 6503 beer-laptop Bottom Expanded Edge Panel\n'
>>>
似乎我的代码错误,只有wmctrl -lp
被执行,| awk '{print $1}'
被省略
我希望输出结果为0x03800081
$ wmctrl -lp | awk '/gedit/ {print $1}'
0x03800081
请帮助。
答案 0 :(得分:7)
使用shell=True
,您应该使用单个命令行而不是数组,否则您的其他参数将被解释为shell参数。来自subprocess
documentation:
在Unix上,shell = True:如果args是一个字符串,它指定要通过shell执行的命令字符串。如果args是一个序列,则第一个项指定命令字符串,任何其他项将被视为附加的shell参数。
所以你的电话应该是:
subprocess.Popen("wmctrl -lp | sed /gedit/ '{print $1}'", shell=True, ...
我认为你的单引号也可能不平衡。
答案 1 :(得分:0)
因为您正在为程序传递序列,所以它认为管道是参数到wmcrtrl
,例如,如果你做了
wmctrl -lp "|"
因此实际的管道操作将丢失。
使它成为单个字符串确实应该给你正确的结果:
>>> import subprocess as s
>>> proc = s.Popen("echo hello | grep e", shell=True, stdout=s.PIPE, stderr=s.PIPE)
>>> proc.stdout.readline()
'hello\n'
>>> proc.stdout.readline()
''
答案 2 :(得分:0)
经过一些研究,我有以下代码,对我来说效果很好。它基本上实时打印stdout和stderr。希望它可以帮助其他需要它的人。
stdout_result = 1
stderr_result = 1
def stdout_thread(pipe):
global stdout_result
while True:
out = pipe.stdout.read(1)
stdout_result = pipe.poll()
if out == '' and stdout_result is not None:
break
if out != '':
sys.stdout.write(out)
sys.stdout.flush()
def stderr_thread(pipe):
global stderr_result
while True:
err = pipe.stderr.read(1)
stderr_result = pipe.poll()
if err == '' and stderr_result is not None:
break
if err != '':
sys.stdout.write(err)
sys.stdout.flush()
def exec_command(command, cwd=None):
if cwd is not None:
print '[' + ' '.join(command) + '] in ' + cwd
else:
print '[' + ' '.join(command) + ']'
p = subprocess.Popen(
command, stdout=subprocess.PIPE, stderr=subprocess.PIPE, cwd=cwd
)
out_thread = threading.Thread(name='stdout_thread', target=stdout_thread, args=(p,))
err_thread = threading.Thread(name='stderr_thread', target=stderr_thread, args=(p,))
err_thread.start()
out_thread.start()
out_thread.join()
err_thread.join()
return stdout_result + stderr_result
如果需要,我认为收集字符串中的输出或错误并返回很容易。