python - 从终端解耦的进程仍然输出 Traceback 到终端
问题描述
在测试我使用 rest api 制作的应用程序时,我发现了这种我不理解的行为。
让我们首先重现一个类似的错误,如下所示 -
在文件中call.py
-
请注意,此文件具有可视化显示自身的代码,例如永远运行的 GUI。在这里,我只是向您展示一个表示,并故意使其引发异常以向您展示问题。发出 get 请求,然后尝试将结果解析为 json 将引发JSONDecodeError
.
import requests
from time import sleep
sleep(3)
uri = 'https://google.com'
r = requests.get(uri)
response_dict = r.json()
由于我想将其作为守护进程运行,因此我使用以下技巧将此进程与启动它的终端分离 -
在文件中start.py
-
import subprocess
import sys
subprocess.Popen(["python3", "call.py"])
sys.exit(0)
然后我执行python3 start.py
它显然解耦了这个过程,因为如果没有例外,视觉表现就会完美运行。
但是,如果出现异常,我会立即在终端中看到此输出,即使我在调用后得到了新提示python3 start.py
-
$ python3 start.py
$ Traceback (most recent call last):
File "call.py", line 7, in <module>
response_dict = r.json()
File "/home/walker/.local/lib/python3.6/site-packages/requests/models.py", line 896, in json
return complexjson.loads(self.text, **kwargs)
File "/usr/lib/python3/dist-packages/simplejson/__init__.py", line 518, in loads
return _default_decoder.decode(s)
File "/usr/lib/python3/dist-packages/simplejson/decoder.py", line 370, in decode
obj, end = self.raw_decode(s)
File "/usr/lib/python3/dist-packages/simplejson/decoder.py", line 400, in raw_decode
return self.scan_once(s, idx=_w(s, idx).end())
simplejson.errors.JSONDecodeError: Expecting value: line 1 column 1 (char 0)
现在,我明白所有异常都必须在程序本身中处理。在这个奇怪的问题之后我已经做到了,但我不清楚的是为什么会发生这种情况?
如果我退出终端并重新启动终端,则不会发生这种情况(在 Traceback 的情况下视觉表现会卡住,并且在任何终端上都没有预期的输出)
为什么解耦进程会以这种方式运行?
注意:解耦对我来说是必不可少的。GUI 必须作为后台或守护进程运行,并且生成它的终端必须从中释放出来。
解决方案
通过“解耦”,我假设您的意思是您希望 stdout/stderr 转到/dev/null
?假设这就是你的意思,那不是你告诉你的代码要做的事情
来自文档:
stdin、stdout 和 stderr 分别指定执行程序的标准输入、标准输出和标准错误文件句柄。有效值为
PIPE
、DEVNULL
、现有文件描述符(正整数)、现有文件对象和None
。使用默认设置
None
,不会发生重定向;子文件句柄将从父文件继承。
因此,您可能想要这样做:
from subprocess import Popen, DEVNULL
Popen(["python3", "call.py"], stdin=DEVNULL, stdout=DEVNULL, stderr=DEVNULL)
根据 OP 的评论,我认为他们可能会使用 GNUscreen
或tmux
. 像这些终端多路复用器,允许您创建一个虚拟终端,您可以在需要时断开和重新连接。这些答案见https://askubuntu.com/a/220880/106239和 https://askubuntu.com/a/8657/106239分别有tmux
和screen
推荐阅读
- python - 如何使用二进制掩码和 STFT 生成音频文件?
- powershell - 在 Pi3 上进行 Windows IOT 远程处理时随机“拒绝访问”
- java - 盒装与原始类型作为实体 id
- linux - 如何从 Linux 中的文件运行 Bash 'export' 命令?
- javascript - 我似乎无法让我的按钮在我的登录屏幕上提交信息
- hibernate - JPA @OneToMany & @ManyToOne 空
- three.js - 将人体 .obj 分割成碎片
- android - 如何将“微调器”放在按钮顶部
- email - SPF 永久错误:DNS 查找过多
- python - 如何放大cartopy