关于subprocess.Popen进程阻塞
背景
在Android系统上实现软件的安装和运行,因为涉及到对返回值的判断,所以经过多方对比后选择了subprocess.Popen()。百度找的常规写法,然后代码运行到一半就阻塞了。实际这行代码是执行了,但是不知道什么原因,资源没有得到及时的释放,然后就阻塞了。
这是阻塞的代码:
一路踩坑
百度后,选择了将输出写入文件中,结束后删除文件的方法,将subprocess.Popen()做以下改进:
def adb_cmd(self,cmd):
'''操作adb shell的方法'''
#得到一个临时文件,在close后删除,
out_temp = tempfile.TemporaryFile()
fileno = out_temp.fileno()
subprocess.Popen(cmd,stdout=fileno,stderr=fileno)
time.sleep(10)
out_temp.seek(0) #指针移到开始处
self.rt = out_temp.readlines()
out_temp.flush()
out_temp.close() #关闭文件,文件就会自动删除
结果还是阻塞,百度过各种方法,因为使用场景不同,都失败。
终于解决
最后还是请教大神。只添加了三行代码,困扰了我多天的问题就解决了。
改进后代码
def adb_cmd(self,cmd):
'''操作adb shell的方法'''
#得到一个临时文件,在close后删除,
out_temp = tempfile.TemporaryFile()
fileno = out_temp.fileno()
res = subprocess.Popen(cmd,stdout=fileno,stderr=fileno)
time.sleep(10)
if res.poll() == None:
res.kill()
res.wait(timeout=30)
out_temp.seek(0) #指针移到开始处
self.rt = out_temp.readlines() #拂去文件
out_temp.flush()
out_temp.close() #关闭文件,文件就会自动删除
刨根问底
虽然问题解决了,但是还是明白原理是什么。避免下次踩坑
百度了一下subprocess.Popen.poll(),用于检查子进程是否已经结束
返回码:
0:正常结束
1:sleep
2:子进程不存在
5:kill
None:在运行
豁然开朗
将每个进程的返回码打印后:
果然是在启动探针时,进程没有正常结束。
终于知道这三行代码的意义