- Flush the standard input with
Ctrl+d - alternatively use
input()orraw_input() - check the docs
>>> help(sys.stdin.read)
- Flush the standard input with
Ctrl+d - alternatively use
input()orraw_input() - check the docs
>>> help(sys.stdin.read)
You should enter the text, press the Enter key, then press Ctrl + Z. It will give the correct result.
For more help, please visit this: https://bugs.python.org/issue5505
[baremetal] sys.stdin does not handle ctrl-D EOF markers
python - What does sys.stdin read? - Stack Overflow
sys.stdin.readline not working as expected
python - sys.stdin.readline() reads without prompt, returning 'nothing in between' - Stack Overflow
Videos
So you have used Python's "pre built in functions", presumably like this:
file_object = open('filename')
for something in file_object:
some stuff here
This reads the file by invoking an iterator on the file object which happens to return the next line from the file.
You could instead use:
file_object = open('filename')
lines = file_object.readlines()
which reads the lines from the current file position into a list.
Now, sys.stdin is just another file object, which happens to be opened by Python before your program starts. What you do with that file object is up to you, but it is not really any different to any other file object, its just that you don't need an open.
for something in sys.stdin:
some stuff here
will iterate through standard input until end-of-file is reached. And so will this:
lines = sys.stdin.readlines()
Your first question is really about different ways of using a file object.
Second, where is it reading from? It is reading from file descriptor 0 (zero). On Windows it is file handle 0 (zero). File descriptor/handle 0 is connected to the console or tty by default, so in effect it is reading from the keyboard. However it can be redirected, often by a shell (like bash or cmd.exe) using syntax like this:
myprog.py < input_file.txt
That alters file descriptor zero to read a file instead of the keyboard. On UNIX or Linux this uses the underlying call dup2(). Read your shell documentation for more information about redirection (or maybe man dup2 if you are brave).
It is reading from the standard input - and it should be provided by the keyboard in the form of stream data.
It is not required to provide a file, however you can use redirection to use a file as standard input.
In Python, the readlines() method reads the entire stream, and then splits it up at the newline character and creates a list of each line.
lines = sys.stdin.readlines()
The above creates a list called lines, where each element will be a line (as determined by the end of line character).
You can read more about this at the input and output section of the Python tutorial.
If you want to prompt the user for input, use the input() method (in Python 2, use raw_input()):
user_input = input('Please enter something: ')
print('You entered: {}'.format(user_input))
I working on a basic password generator and am running the following function to check the password length based on user input.
def passLength():
len = sys.stdin.readline()
try:
len = int(len)
except ValueError:
print(' Must enter a number')
len = None
passLength()
len = int(len)
if len < 6:
print(' Invalid Entry. Password must be at least 6 characters in length')
passLength()
else:
return lenThis works... unless the user types alphabetic characters prior to a number. Through debugging I found that "sys.stdin.readline" isn't adding new values to the "len" variable when the function is called multiple times.
You'll notice I'm clearing "len" by setting it to "None" I have also attempted to resolve it by setting "len" to an empty string with no success.
What can I do to either 1) fully reset my passLength function or 2) effectively clear previous data from "len" so "sys.stdin.readline" can actually add new data.
GitHub Repo here with full code
stdin.read(1) reads one character from stdin. If there was more than one character to be read at that point (e.g. the newline that followed the one character that was read in) then that character or characters will still be in the buffer waiting for the next read() or readline().
As an example, given rd.py:
from sys import stdin
x = stdin.read(1)
userinput = stdin.readline()
betAmount = int(userinput)
print ("x=",x)
print ("userinput=",userinput)
print ("betAmount=",betAmount)
... if I run this script as follows (I've typed in the 234):
C:\>python rd.py
234
x= 2
userinput= 34
betAmount= 34
... so the 2 is being picked up first, leaving the 34 and the trailing newline character to be picked up by the readline().
I'd suggest fixing the problem by using readline() rather than read() under most circumstances.
Simon's answer and Volcano's together explain what you're doing wrong, and Simon explains how you can fix it by redesigning your interface.
But if you really need to read 1 character, and then later read 1 line, you can do that. It's not trivial, and it's different on Windows vs. everything else.
There are actually three cases: a Unix tty, a Windows DOS prompt, or a regular file (redirected file/pipe) on either platform. And you have to handle them differently.
First, to check if stdin is a tty (both Windows and Unix varieties), you just call sys.stdin.isatty(). That part is cross-platform.
For the non-tty case, it's easy. It may actually just work. If it doesn't, you can just read from the unbuffered object underneath sys.stdin. In Python 3, this just means sys.stdin.buffer.raw.read(1) and sys.stdin.buffer.raw.readline(). However, this will get you encoded bytes, rather than strings, so you will need to call .decode(sys.stdin.decoding) on the results; you can wrap that all up in a function.
For the tty case on Windows, however, input will still be line buffered even on the raw buffer. The only way around this is to use the Console I/O functions instead of normal file I/O. So, instead of stdin.read(1), you do msvcrt.getwch().
For the tty case on Unix, you have to set the terminal to raw mode instead of the usual line-discipline mode. Once you do that, you can use the same sys.stdin.buffer.read(1), etc., and it will just work. If you're willing to do that permanently (until the end of your script), it's easy, with the tty.setraw function. If you want to return to line-discipline mode later, you'll need to use the termios module. This looks scary, but if you just stash the results of termios.tcgetattr(sys.stdin.fileno()) before calling setraw, then do termios.tcsetattr(sys.stdin.fileno(), TCSAFLUSH, stash), you don't have to learn what all those fiddly bits mean.
On both platforms, mixing console I/O and raw terminal mode is painful. You definitely can't use the sys.stdin buffer if you've ever done any console/raw reading; you can only use sys.stdin.buffer.raw. You could always replace readline by reading character by character until you get a newline… but if the user tries to edit his entry by using backspace, arrows, emacs-style command keys, etc., you're going to get all those as raw keypresses, which you don't want to deal with.
Use the fileinput module:
import fileinput
for line in fileinput.input():
pass
fileinput will loop through all the lines in the input specified as file names given in command-line arguments, or the standard input if no arguments are provided.
Note: line will contain a trailing newline; to remove it use line.rstrip().
There's a few ways to do it.
sys.stdinis a file-like object on which you can call functionsreadorreadlinesif you want to read everything or you want to read everything and split it by newline automatically. (You need toimport sysfor this to work.)If you want to prompt the user for input, you can use
raw_inputin Python 2.X, and justinputin Python 3.If you actually just want to read command-line options, you can access them via the sys.argv list.
You will probably find this Wikibook article on I/O in Python to be a useful reference as well.
Hey everyone, I just want to talk about reading in data from standard input and the 4 main ways it can be done.
I'm not going to talk about the input() or raw_input() functions today, instead ill be talking about how to read from standard input using the sys module.
To get access to the sys module we first need to import it
import sys
Ok now we have access to this module, there are 3 ways to read from standard input:
-
sys.stdin.read([size])
-
sys.stdin.readline()
-
sys.stdin.readlines()
Lets look at how all of these work first and the ways to use them.
First off we can read lines directly from the console, this will look something like this
lines = sys.stdin.read() print(lines) $ python3 stdin.py Line1 Line 2 **END** Line 1 Line 2
Our lines variable looks like this: "Line1\nLine2"
Here when we run our program, it waits until it we pass some data through the console window. We specify end of input using ctrl+z on windows and I believe ctrl+d on linux.
The sys.stdin.read() function also has an optional parameter for the size of the data we want to read. For example if we pass 10 then it reads 10 characters including any newline characters.
The read() function will read everything, or the size of data specified, and return it as one string. This is useful for small amounts of data but if we read large files this way, it can use up a lot of memory.
The second way is sys.stdin.readline() which is self explanatory and reads a single line from standard input with a newline character at the end.
line = sys.stdin.readline() print(line) $ python3 stdin.py hello hello
The next way is sys.stdin.readlines(). I find myself using this way most often. With this way, we read lines from the console and are returned a list containing all the lines we entered.
lines = sys.stdin.readlines() print(lines) $ python3 stdin.py line1 line2 line3 ['line1\n', 'line2\n', 'line3\n']
This is very useful if we wish to process a file line by line although, we do have a large list sitting in memory which we may not want with large files. I will show you how to read from files in a moment.
Reading from files:
To read from a file we can do this a couple of ways, we can open and read the file within our program.
with open('FILENAME', [rw]) as our_file:
for line in our_file:
print(line)The optional [rw] specifies whether we wish to open the file for reading, r or writing, w. This will work depending on the access permission on the file. You can check this on linux from the command line by navigating to your directory where the file is and typing:
$ ls -l
This will display the access permissions of the file in that directory.
An error will be thrown if you try to read or write without having permission to do so.
If the file name you entered doesn't exist, an empty file will be created for you.
The use of with open() here is very useful as it closes our file for us when we are finished.
Another way to read a file is passing it at the command line
$ python3 stdin.py < FILENAME.txt
Presuming FILENAME.txt looks like this:
Line 1 Line 2 Line 3
Running the following program, we get the following output:
import sys lines = sys.stdin.readlines() print(lines) $ python3 stdin.py < FILENAME.txt ['Line 1\n', 'Line 2\n', 'Line 3']
I dont want to talk to much about the different ways of reading and writing files as I only wanted to talk about the different methods we have available to use for reading so I wont discuss any further ways of reading.
If we wish to strip the newline characters from our lines we can use the strip() method, I'm going to use a list comprehension here as it is a good example of their usage:
lines = [line.strip() for line in sys.stdin.readlines()] print(lines) $ python3 stdin.py < FILENAME.txt ['Line 1', Line 2', 'Line 3']
Whats the list comprehension doing? It uses a for loop to loop through each line in standard input, takes each line and strips it then appends it to our list, lines.
Now our newline characters are gone.
We covered a fair bit of stuff here and got the chance to see some extra things in use such as list comprehensions. If you found anything here confusing, play around with it yourself, after all its one of the best ways to learn.
Problem is probably due to flushing of stdin since the \n lingers on.
as an alternative, use raw_input
while True:
c = raw_input('please enter a character: ')
print 'you entered', c
For the flushing part, see this
sys.stdin is line-buffered by default i.e., your sys.stdin.read(1) won't return until there is full line in stdin buffer.
It means if you enter a character and hit Enter then after you get the first character with sys.stdin.read(1), there is a newline in the buffer (one or two characters: os.linesep) that are read immediately on the next loop iterations.
You could avoid hitting Enter by reading exactly one character at a time (msvcrt.getch()).
@Adar70
EOF is not a character you can send, it's a state/condition, which is true when end of file is detected. For a process sending the data (as opposed to reading from a file) that requires either to terminate/exit the sending process (your QProcess) --- which will mean it won't be there to read the response back in your case --- or you must close stdout, QProcess::closeWriteChannel() --- which will mean you won't be able to send any further messages to the subprocess, which may or may not be acceptable in your situation.
However, better, I will bet my bottom dollar that Python has a way of reading from stdin which does not require EOF. That is down to your statements like sys.stdin.read(), which is a Python question. You want some kind of "unbuffered" and "return immediately with what is there". That is for you to look up. I don't think you should read from sys.stdin.buffer, that is likely to be buffered. Why don't you start with os.read(0, 5) --- read 5 bytes from file descriptor 0, which is stdin, and should be unbuffered --- and see if that works?
Failing that you might try sending a \n at the end of your message, if Python's stdin is line-buffered that may work. If you had sent the data from Pythion via print(..., flush=True) that would have added a \n and I believe that would allow reader to use e.g. for line in sys.stdin (or equivalent for one line) and receive the line immediately. However, your data looks like arbitrary binary data, not text, so the whole idea of inserting a \n or reading "lines" may not be appropriate here.
You might also temporarily try writing the sender side in Python too while you get it working.
The thing is you call start and right after your write something to the stdin however there's no guarantee that your script has started at this point.
You should use the started signal to then write to the process and you should also connect the errorOccurred signal because you never know what might go wrong.
in Python the
EOFError: EOF when reading a line
there is 2 reason for this error
1.reading the file in the wrong way/format
import sys
for line in sys.stdin:
print (line)
this how we can read using "sys.stdln"
2.there is another chance for the same error if the file is corrupted
Reading from stdin after Ctrl-D is normally allowed, but I have tested this only on Ubuntu (code similar to yours works perfectly well). I see that this is being run on Windows and the Windows console might behave differently and refuse any read() operations after Ctrl-D. One possible solution is to capture the EOFError exception with a try/except statement and close and re-open sys.stdin when the exception occurs. Something like this:
# note: check that sys.stdin.isatty() is True!
try:
# read/input here
except EOFError:
sys.stdin.close()
sys.stdin = open("con","r")
continue # or whatever you need to do to repeat the input cycle