如何将字节(u8)的矢量转换为字符串?

我试图在Rust中编写简单的TCP/IP客户端,我需要打印出我从服务器获得的缓冲区。

如何将Vec<u8>(或&[u8])转换为String?

185541 次浏览

将字节片转换为字符串片(假设为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

如果您确定字节片是有效的UTF-8,并且不想招致有效性检查的开销,则存在此函数的不安全版本from_utf8_unchecked,它具有相同的行为,但会跳过检查。

如果你需要一个String而不是&str,你也可以考虑String::from_utf8代替。

转换函数的标准库引用:

我更喜欢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()方法,因为它是在写时复制的。

如果你实际上有一个字节向量 (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()

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解码输出的新字符串分配空间。