我试图在Rust中编写简单的TCP/IP客户端,我需要打印出我从服务器获得的缓冲区。
如何将Vec<u8>(或&[u8])转换为String?
Vec<u8>
&[u8]
String
将字节片转换为字符串片(假设为UTF-8编码):
use std::str; // // pub fn from_utf8(v: &[u8]) -> Result<&str, Utf8Error> // // Assuming buf: &[u8] // fn main() { let buf = &[0x41u8, 0x41u8, 0x42u8]; let s = match str::from_utf8(buf) { Ok(v) => v, Err(e) => panic!("Invalid UTF-8 sequence: {}", e), }; println!("result: {}", s); }
转换是就地进行的,不需要分配。如果需要,可以通过调用字符串片上的.to_owned() (还有其他选择)从字符串片创建String。
.to_owned()
如果您确定字节片是有效的UTF-8,并且不想招致有效性检查的开销,则存在此函数的不安全版本from_utf8_unchecked,它具有相同的行为,但会跳过检查。
from_utf8_unchecked
如果你需要一个String而不是&str,你也可以考虑String::from_utf8代替。
String::from_utf8
转换函数的标准库引用:
std::str::from_utf8
std::str::from_utf8_unchecked
std::string::String::from_utf8
我更喜欢String::from_utf8_lossy:
String::from_utf8_lossy
fn main() { let buf = &[0x41u8, 0x41u8, 0x42u8]; let s = String::from_utf8_lossy(buf); println!("result: {}", s); }
它将无效的UTF-8字节转换为-,因此不需要任何错误处理。当你不需要它的时候它很好,而我几乎不需要它。你实际上得到了一个String。它应该使您更容易打印出从服务器获取的内容。
有时你可能需要使用into_owned()方法,因为它是在写时复制的。
into_owned()
如果你实际上有一个字节向量 (Vec<u8>),并且想要转换为String,最有效的方法是使用String::from_utf8重用分配:
fn main() { let bytes = vec![0x41, 0x42, 0x43]; let s = String::from_utf8(bytes).expect("Found invalid UTF-8"); println!("{}", s); }
在我的例子中,我只需要将数字转换为字符串,而不是根据某种编码将数字转换为字母,所以我这样做了
fn main() { let bytes = vec![0x41, 0x42, 0x43]; let s = format!("{:?}", &bytes); println!("{}", s); }
为了最优地将可能包含非UTF-8字符/字节序列的Vec<u8>转换为UTF-8 String 没有任何不需要的分配,您将乐观地尝试调用String::from_utf8(),然后求助于String::from_utf8_lossy()。
String::from_utf8()
String::from_utf8_lossy()
let buffer: Vec<u8> = ...; let utf8_string = String::from_utf8(buffer) .map_err(|non_utf8| String::from_utf8_lossy(non_utf8.as_bytes()).into_owned()) .unwrap();
其他答案中建议的方法将导致在内存中拥有两个缓冲区,即使在愉快的情况下(向量中有有效的UTF-8数据):一个是原始的u8字节,另一个是String拥有其字符的形式。此方法将尝试使用Vec<u8>并直接将其编组为Unicode String,只有失败时才会为包含有损的UTF-8解码输出的新字符串分配空间。
u8