如何从bash脚本向Python发送SIGINT?

Rya*_*yan 9 python bash

我想从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'
Run Code Online (Sandbox Code Playgroud)

从shell我可以打断它:

$ python loop.py &
[1] 15420
starting loop
$ kill -SIGINT 15420
quitting loop
[1]+  Done                    python loop.py
Run Code Online (Sandbox Code Playgroud)

kill.sh:

#! /bin/bash
python loop.py &
PID=$!
echo "sending SIGINT to process $PID"
kill -SIGINT $PID
Run Code Online (Sandbox Code Playgroud)

但是从脚本我不能:

$ ./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
Run Code Online (Sandbox Code Playgroud)

并且,如果它是从脚本启动的,我就不能再从shell中删除它了:

$ kill -SIGINT 15452
$ ps ax | grep loop.py | grep -v grep
15452 pts/3    R      0:34 python loop.py
Run Code Online (Sandbox Code Playgroud)

我假设我错过了一些关于bash工作控制的好点.

Mat*_*hen 15

你没有注册信号处理程序.试试下面的内容.它看起来相当可靠.我认为罕见的例外是它在Python注册脚本的处理程序之前捕获信号.请注意,只有当用户点击中断键时才会引发KeyboardInterrupt.我认为它适用于显式(例如,通过杀死)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()
Run Code Online (Sandbox Code Playgroud)


Ste*_*een 5

除了@matthew-flaschen 的回答之外,您还可以exec在 bash 脚本中使用以有效地将范围替换为正在打开的进程:

#!/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
Run Code Online (Sandbox Code Playgroud)