標籤:
下面就是今天下午的研究成果。 發布系統需要響應使用者的插斷要求,需要在GET方法中殺掉由subprocess派生的子進程,剛開始直接用os.kill 發現子進程的子進程無法kill,Google了一些,發現kill可以幹掉進程組,於是測試,但是預設情況下,subprocess派生的進程組和主程式,也就是我的web.py進程是在一個進程組裡的,這要是kill了,那就調的了。 繼續翻google,看subprocess的document時發現這個變數:subprocess.CREATE_NEW_PROCESS_GROUP
A Popen creationflags parameter to specify that a new process group will be created. This flag is necessary for using os.kill() on the subprocess.
This flag is ignored if CREATE_NEW_CONSOLE is specified.
比較高興,以為能解決問題了,結果測試半天,才瞭解這玩意是only windows的,我去啊,不過想到了,win能做到的,linux肯定也可以,於是定位到
preexec_fn
又是一通google,不是對象嗎,弄了個setpgid(0,0) 測試了,子進程還是和主調進程屬於同一個進程組,後來靈機一動:
preexec_fn = os.setpgrp 這樣竟然解決了新產生進程組的問題。繼續努力,後面遇到的就是僵死進程的問題了,os.waitpid了一下就解決了。剛開始waitpid的時候,還在linxu上man了半天,看著linxu手冊裡的參數,還是不放心啊,結果python裡的os.waitpid竟然沒有那麼多參數,而且沒有傳回值,簡陋啊。不過正解決了我的問題。 下面是今天的完全測試代碼。 [[email protected] kill-subprocess]$ cat sub-process.py import subprocessimport osimport time def my_func():#派生兩個子進程,子進程裡又派生幾個sleep的孫子進程,主要是為了測試kill進程組。 run_str2 = ‘/bin/sh test.sh‘ run_str = ‘/bin/sh test_quick.sh‘ cmd2 = run_str.split() cmd = run_str.split()#測試了一些個preexec_fn的值,最終發現能用的,對python的對象的概念還是不理解啊,新手,新手。 #p = subprocess.Popen(cmd, stdin = subprocess.PIPE, stdout = subprocess.PIPE, stderr = subprocess.PIPE, shell = False, creationflags = subprocess.CREATE_NEW_PROCESS_GROUP) #p = subprocess.Popen(cmd, stdin = subprocess.PIPE, stdout = subprocess.PIPE, stderr = subprocess.PIPE, shell = False, creationflags = 0) p = subprocess.Popen(cmd, stdin = subprocess.PIPE, stdout = subprocess.PIPE, stderr = subprocess.PIPE, shell = False, preexec_fn = os.setpgrp ) p2 = subprocess.Popen(cmd2, stdin = subprocess.PIPE, stdout = subprocess.PIPE, stderr = subprocess.PIPE, shell = False, preexec_fn = os.setpgrp ) #@p = subprocess.Popen(cmd, stdin = subprocess.PIPE, stdout = subprocess.PIPE, stderr = subprocess.PIPE, shell = False, preexec_fn = os.setpgid(0, 0) ) pid = p.pid pgid = os.getpgid(pid) print "pid: %d\n" %pid print "pgid: %d\n" %pgid return pid pid = my_func()#p.wait()print "now , sleep 2s ,then , os.kill gpid %d" % pidtime.sleep(20) a = os.kill(-pid, 9)print "kill,return:"print a # kill的時候,我測試了kill 沒有許可權的root進程,會報錯:許可權不允許# 測試了kill p p2 都可以kill#a = os.kill(2445, 9)#print "kill root process 2445 ,return:"#print a#p.wait()#os.waitpid(pgid, 0)# 2445 is a root process#os.waitpid(2445, 0)#os.waitpid(p2.pid, 0)os.waitpid(pid, 0)print "waitpid,return:"print atime.sleep(22) print "done..." #p.terminate()#p.kill()#p.wait()##time.sleep(40)#os.kill(pid, 9)
python subprocess 殺掉全部派生的子進程