我在一个简单的gui应用程序上工作。
在我的实际案例中,这个应用程序启动Django( python manage.py runserver )webserver并关闭Django webserver。为了简化我的应用程序的可测试性,我通过 sleep 命令替换django start webserver。
当我触发启动服务器按钮时:服务器启动。 然后,我触发停止服务器按钮,然后关闭该过程。
但是django webserver(或sleep命令)仍在工作。
如何停止 subprocess.Popen 调用的进程并停止de命令?
有我的代码:
#!/usr/bin/python
# coding:utf-8
import os
import sys
import subprocess
import time
import datetime
import zipfile
import subprocess
from concurrent import futures
from threading import Thread
from multiprocessing import Process
try:
import Tkinter as tk
except ImportError:
import tkinter as tk
def runserver():
cmd = 'sleep 60'
# os.system(cmd)
subprocess.Popen(cmd, shell=True, stdout=subprocess.PIPE).stdout.read()
print('TERMINATE BY TIMEOUT')
return True
class WindowMain:
def __init__(self, master):
self.thread_srv = None
self.master = master
self.frame = tk.Frame(self.master)
self.lbl = tk.Label(text="")
self.lbl.pack()
self.btn_quit = tk.Button(self.frame, text="Stop server", width=25, command=lambda: self.stop_server())
self.btn_quit.pack()
self.button1 = tk.Button(self.frame, text="Start Server", width=25, command=lambda: self.runserver_thread())
self.button1.pack()
self.frame.pack()
def stop_server(self):
if self.thread_srv:
print('THREAD')
self.thread_srv.terminate()
else:
print("NOT THREAD")
def runserver_thread(self):
self.thread_srv = Process(target=runserver)
self.thread_srv.start()
current_dir = os.path.dirname(os.path.abspath(__file__))
def quit(root):
root.quit
sys.exit()
def main():
root = tk.Tk()
tk.Button(root, text="Quitter", command=lambda: quit(root)).pack()
app = WindowMain(root)
root.mainloop()
if __name__ == '__main__':
main()
我编写了关闭django服务器的这个函数:
def close_django_runserver():
if 'win' in sys.platform:
pass
else:
cmd = 'ps aux'
output = subprocess.Popen(cmd, shell=True, stdout=subprocess.PIPE).stdout.read()
process_list = output.split("\n")
for process in process_list:
if 'manage.py' in process and 'runserver' in process:
pid = re.sub(r'^[^\s]+\s+(\d+)\s+.*$', r'\1', process)
subprocess.Popen('kill '+str(pid), shell=True, stdout=subprocess.PIPE).stdout.read()
它确实存在更好的解决方案,无论如何。我将在稍后编写Windows解决方案。
答案 0 :(得分:2)
你运行Django启动脚本,所有内容都被初始化并设置,然后新创建的Django进程转到后台(获取* daemonized *)与线程断开连接。
所以,当你杀死那个线程时,你实际上什么也没杀,因为启动脚本完成了它的工作并且可能已经退出了。
运行启动脚本后,将创建新的Django进程。在您的机器上进行实验并找到它的名字。然后,您可以在程序中运行类似sudo killall django_process
的内容。