关于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:在运行
豁然开朗
将每个进程的返回码打印后:

果然是在启动探针时,进程没有正常结束。
终于知道这三行代码的意义
这篇博客讲述了作者在Android系统上使用subprocess.Popen()进行进程操作时遇到的阻塞问题。尝试了多种解决方案,如将输出写入文件、查询子进程状态等,最终通过添加三行代码成功解决问题,并探讨了subprocess.Popen.poll()的作用,了解了进程结束的状态码,以防止未来再次遇到类似问题。
1566

被折叠的 条评论
为什么被折叠?



