sys.stdin 和 input() 之间的冲突 - EOFError:读取一行时出现 EOF

rme*_*ier 5 python stdin python-3.x

我无法在不引发 EOFError 异常的情况下使以下脚本工作:

#!/usr/bin/env python3

import json
import sys

# usage:
# echo '[{"testname": "testval"}]' | python3 test.py

myjson = json.load(sys.stdin)
print(json.dumps(myjson))

answer = input("> ")  # BUG: EOFError: EOF when reading a line
print(answer)
Run Code Online (Sandbox Code Playgroud)

我读过这个似乎相关的问题:Python STDIN User Input Issue

我认为这告诉我我需要清除标准输入缓冲区?但我不确定如何,因为print(sys.stdin.readline())只是输出一个换行符而 EOFError 仍然存在。

我也尝试使用该sys.stdin.flush()方法(在这个问题中找到:Usage of sys.stdout.flush() method)虽然我仍然不明白它的作用,因为我在官方文档(3.6)中找不到它,最接近我发现是这个,但它没有提到刷新:https : //docs.python.org/3/library/sys.html

请记住,我不是程序员,也没有 CS 教育或背景。我只是编写脚本来自动化我的部分非技术工作。因此,如果您知道关于标准输入/标准输出如何在 Python 的 shell 中工作的任何好的初学者资源,请务必告诉。

jed*_*rds 5

通过管道输入,Python 将 sys.stdin 作为 FIFO 打开。否则,Python 将打开 sys.stdin/dev/tty或等效文件。

您可以通过以下方式验证:

import os,sys,stat
print("isatty():", sys.stdin.isatty())
print("isfifo():", stat.S_ISFIFO(os.fstat(0).st_mode))
Run Code Online (Sandbox Code Playgroud)

运行两次,一次输入数据,一次不输入。

我得到:

$ echo "Test" | ./test2.py
isatty(): False
isfifo(): True

$ ./test2.py
isatty(): True
isfifo(): False
Run Code Online (Sandbox Code Playgroud)

所以你的 EOF 发生是因为 FIFO sys.stdin 被打开空。

但是,您可以将 sys.stdin 重新打开到/dev/tty

j = json.load(sys.stdin)
print(j)

sys.stdin = open("/dev/tty")

answer = input("> ")
print(answer)
Run Code Online (Sandbox Code Playgroud)

哪个可以正常工作:

$ echo '{"key":"val"}' | python3 ./testjson.py
{'key': 'val'}
> testing
testing
Run Code Online (Sandbox Code Playgroud)