我想运行一个在stdin上阻塞的可执行文件,当按下某个键时,不会立即打印相同的字符而不必按 Enter 。
如何在不必点击 Enter 的情况下从stdin中读取一个字符?我从这个例子开始:
fn main() {
println!("Type something!");
let mut line = String::new();
let input = std::io::stdin().read_line(&mut line).expect("Failed to read line");
println!("{}", input);
}
我查看了API并尝试将read_line()
替换为bytes()
,但我尝试的所有操作都要求我在读取之前按 Enter 。
这个问题被要求提供C / C ++,但似乎没有标准的方法:Capture characters from standard input without waiting for enter to be pressed
考虑到它在C / C ++中并不简单,在Rust中可能不可行。
答案 0 :(得分:11)
使用现在可用的“ncurses”库之一,例如this一个。
在货物中添加依赖
[dependencies]
ncurses = "5.86.0"
并包含在main.rs中:
extern crate ncurses;
use ncurses::*; // watch for globs
按照库中的示例初始化ncurses并等待单个字符输入,如下所示:
initscr();
/* Print to the back buffer. */
printw("Hello, world!");
/* Update the screen. */
refresh();
/* Wait for a key press. */
getch();
/* Terminate ncurses. */
endwin();
答案 1 :(得分:11)
虽然@ Jon的解决方案使用ncurses工作,但ncurses按设计清除屏幕。我想出了这个解决方案,它使用termios crate为我的小项目学习Rust。我们的想法是通过termios绑定访问ECHO
来修改ICANON
和tcsetattr
标记。
extern crate termios;
use std::io;
use std::io::Read;
use std::io::Write;
use termios::{Termios, TCSANOW, ECHO, ICANON, tcsetattr};
fn main() {
let stdin = 0; // couldn't get std::os::unix::io::FromRawFd to work
// on /dev/stdin or /dev/tty
let termios = Termios::from_fd(stdin).unwrap();
let mut new_termios = termios.clone(); // make a mutable copy of termios
// that we will modify
new_termios.c_lflag &= !(ICANON | ECHO); // no echo and canonical mode
tcsetattr(stdin, TCSANOW, &mut new_termios).unwrap();
let stdout = io::stdout();
let mut reader = io::stdin();
let mut buffer = [0;1]; // read exactly one byte
print!("Hit a key! ");
stdout.lock().flush().unwrap();
reader.read_exact(&mut buffer).unwrap();
println!("You have hit: {:?}", buffer);
tcsetattr(stdin, TCSANOW, & termios).unwrap(); // reset the stdin to
// original termios data
}
读取单个字节的一个优点是捕获箭头键,ctrl等。不会捕获扩展的F键(尽管ncurses可以捕获这些)。
此解决方案适用于类UNIX平台。我没有使用Windows的经验,但根据这个forum,在Windows中使用SetConsoleMode
可能会有类似的东西。
答案 2 :(得分:1)
您也可以使用termion,但是您必须启用raw TTY mode,这也会改变stdout
的行为。请参见下面的示例(经过Rust 1.34.0测试)。请注意,在内部,它还包装termios UNIX API。
[dependencies]
termion = "1.5.2"
use std::io;
use std::io::Write;
use std::thread;
use std::time;
use termion;
use termion::input::TermRead;
use termion::raw::IntoRawMode;
fn main() {
// Set terminal to raw mode to allow reading stdin one key at a time
let mut stdout = io::stdout().into_raw_mode().unwrap();
// Use asynchronous stdin
let mut stdin = termion::async_stdin().keys();
loop {
// Read input (if any)
let input = stdin.next();
// If a key was pressed
if let Some(Ok(key)) = input {
match key {
// Exit if 'q' is pressed
termion::event::Key::Char('q') => break,
// Else print the pressed key
_ => {
write!(
stdout,
"{}{}Key pressed: {:?}",
termion::clear::All,
termion::cursor::Goto(1, 1),
key
)
.unwrap();
stdout.lock().flush().unwrap();
}
}
}
thread::sleep(time::Duration::from_millis(50));
}
}