【Mio】TCP 服务器(七)
环境
- Time 2022-01-08
- Rust 1.57.0
- mio 0.8
概念
参考:https://github.com/tokio-rs/mio/blob/master/examples/tcp_server.rs
接受客户端发送的数据。
示例
读取数据
if event.is_readable() {
println!("状态可读");
let mut buffer = vec![0; 4096];
let mut bytes_read = 0;
loop {
match client.read(&mut buffer) {
Ok(n) => bytes_read = n,
Err(e) if e.kind() == WouldBlock => break,
Err(err) => return Err(err),
}
}
if bytes_read != 0 {
let received = &buffer[..bytes_read];
let str = from_utf8(received).unwrap();
println!("收到数据:{}", str.trim_end());
}
};
使用 nc 命令访问
客户端: 127.0.0.1:50716
状态可写
状态可读
收到数据:4444
状态可写
状态可读
收到数据:你好
状态可写
总结
读取客户端发送的数据。
附录
完整代码
use mio::{
event::Event,
net::{TcpListener, TcpStream},
Events, Interest, Poll, Token,
};
use std::{
collections::HashMap,
io::{self, ErrorKind::WouldBlock, Read, Write},
str::from_utf8,
};
const SERVER: Token = Token(0);
fn main() -> io::Result<()> {
let addr = "127.0.0.1:4444".parse().unwrap();
let mut server = TcpListener::bind(addr)?;
let mut poll = Poll::new()?;
let mut events = Events::with_capacity(128);
poll.registry()
.register(&mut server, SERVER, Interest::READABLE)?;
let mut clients = HashMap::new();
let mut unique_token = SERVER;
loop {
poll.poll(&mut events, None)?;
for event in events.iter() {
match event.token() {
SERVER => {
while let Some((token, client)) = accept(&server, &mut unique_token, &poll)? {
clients.insert(token, client);
}
}
token => {
if let Some(client) = clients.get_mut(&token) {
handle_client(event, client)?;
}
}
}
}
}
}
fn accept(
server: &TcpListener,
unique_token: &mut Token,
poll: &Poll,
) -> io::Result<Option<(Token, TcpStream)>> {
let (mut client, address) = match server.accept() {
Ok((client, address)) => (client, address),
Err(e) if e.kind() == WouldBlock => return Ok(None),
Err(e) => return Err(e),
};
unique_token.0 += 1;
let token = *unique_token;
poll.registry().register(
&mut client,
token,
Interest::READABLE.add(Interest::WRITABLE),
)?;
println!("客户端: {}", address);
Ok(Some((token, client)))
}
fn handle_client(event: &Event, client: &mut TcpStream) -> io::Result<()> {
if event.is_writable() {
println!("状态可写");
client.write_all(b"hello world\n").unwrap();
}
if event.is_readable() {
println!("状态可读");
let mut buffer = vec![0; 4096];
let mut bytes_read = 0;
loop {
match client.read(&mut buffer) {
Ok(n) => bytes_read = n,
Err(e) if e.kind() == WouldBlock => break,
Err(err) => return Err(err),
}
}
if bytes_read != 0 {
let received = &buffer[..bytes_read];
let str = from_utf8(received).unwrap();
println!("收到数据:{}", str.trim_end());
}
};
Ok(())
}
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 阿里最新开源QwQ-32B,效果媲美deepseek-r1满血版,部署成本又又又降低了!
· 单线程的Redis速度为什么快?
· SQL Server 2025 AI相关能力初探
· AI编程工具终极对决:字节Trae VS Cursor,谁才是开发者新宠?
· 展开说说关于C#中ORM框架的用法!