我想使用 mio crate 来读取按键,因为它们以无缓冲的方式到达。我已经有了取消缓冲标准输入的代码,并且已经搭建了事件循环的脚手架:
extern crate mio;
extern crate termios;
use termios::{Termios, TCSANOW, ICANON, ECHO, tcsetattr};
use mio::*;
use mio::unix::EventedFd;
fn unbuffer_stdin() {
let termios = Termios::from_fd(0).unwrap();
let mut new_termios = termios.clone();
new_termios.c_lflag &= !(ICANON | ECHO);
tcsetattr(0, TCSANOW, &mut new_termios).unwrap();
}
fn main() {
let stdin = 0;
unbuffer_stdin();
let poll = Poll::new().unwrap();
const STDIN: Token = Token(0);
let ev_fd = EventedFd(&stdin);
poll.register(&ev_fd, STDIN, Ready::readable(), PollOpt::edge()).unwrap();
let mut events = Events::with_capacity(1024);
loop {
poll.poll(&mut events, None).unwrap();
for event in events.iter() {
match event.token() {
STDIN => {
println!("keypress");
// XXX read in ready codepoints to a buffer
}
_ => unreachable!(),
}
}
}
}
如何实现标记的部分XXX
?有几个挑战:
- 我怎么知道要读取多少字节?我不确定mio会告诉我这个。
- 如何处理部分代码点,其中读取跨越多字节字符。
一种可行的解决方案是使用oneshot
事件而不是edge
事件,然后将每个事件的一个字节读入临时缓冲区。每次缓冲区作为一个完整的代码点有意义时,我都可以将其转换为 achar
并将其存储起来,并清除暂存缓冲区。
不过,这似乎有点低效。什么是最好的方法?