我想从bash脚本启动一个后台Python作业,然后使用SIGINT优雅地终止它。这在shell中工作得很好,但是我似乎无法在脚本中让它工作。
loop.py:
#! /usr/bin/env python
if __name__ == "__main__":
try:
print 'starting loop'
while True:
pass
except KeyboardInterrupt:
print 'quitting loop'在shell中我可以中断它:
$ python loop.py &
[1] 15420
starting loop
$ kill -SIGINT 15420
quitting loop
[1]+ Done python loop.pykill.sh:
#! /bin/bash
python loop.py &
PID=$!
echo "sending SIGINT to process $PID"
kill -SIGINT $PID但是从一个脚本中我不能:
$ ./kill.sh
starting loop
sending SIGINT to process 15452
$ ps ax | grep loop.py | grep -v grep
15452 pts/3 R 0:08 python loop.py而且,如果它是从脚本启动的,我就不能再从shell中删除它:
$ kill -SIGINT 15452
$ ps ax | grep loop.py | grep -v grep
15452 pts/3 R 0:34 python loop.py我想我漏掉了一些bash任务控制的细节。
发布于 2009-06-10 08:00:35
您没有注册信号处理程序。试试下面的方法。它似乎工作得相当可靠。我认为罕见的例外是在Python注册脚本的处理程序之前捕获信号。请注意,“当用户按下中断键时”,才应该引发KeyboardInterrupt。我认为它在显式(例如,通过kill) SIGINT中工作的事实是一个实现上的意外。
import signal
def quit_gracefully(*args):
print 'quitting loop'
exit(0);
if __name__ == "__main__":
signal.signal(signal.SIGINT, quit_gracefully)
try:
print 'starting loop'
while True:
pass
except KeyboardInterrupt:
quit_gracefully()发布于 2017-06-14 15:56:36
除了@matthew-flaschen的回答之外,您还可以在bash脚本中使用exec来有效地替换正在打开的进程的作用域:
#!/bin/bash
exec python loop.py &
PID=$!
sleep 5 # waiting for the python process to come up
echo "sending SIGINT to process $PID"
kill -SIGINT $PID发布于 2009-06-10 08:12:14
我同意Matthew Flaschen的观点;问题出在python,当它不是从交互式shell调用时,它显然不会向SIGINT注册KeyboardInterrupt异常。
当然,没有什么可以阻止您这样注册您的信号处理程序:
def signal_handler(signum, frame):
raise KeyboardInterrupt, "Signal handler"https://stackoverflow.com/questions/974189
复制相似问题