如何连接以下类型组合:
str 和 str
字符串和字符串
字符串和字符串
str
和 &str
是不同的类型,在 99% 的情况下,您只需关心 &str
。还有其他问题详细说明了它们之间的差异。
连接字符串时,需要分配内存来存储结果。最容易开始的是 String
和 &str
:
fn main() {
let mut owned_string: String = "hello ".to_owned();
let borrowed_string: &str = "world";
owned_string.push_str(borrowed_string);
println!("{}", owned_string);
}
在这里,我们有一个可以变异的自有字符串。这是有效的,因为它可能允许我们重用内存分配。 String
和 String
的情况与 &String
can be dereferenced as &str
类似。
fn main() {
let mut owned_string: String = "hello ".to_owned();
let another_owned_string: String = "world".to_owned();
owned_string.push_str(&another_owned_string);
println!("{}", owned_string);
}
在此之后,another_owned_string
保持不变(注意没有 mut
限定符)。还有另一个变体使用 String
但不要求它是可变的。这是一个 implementation of the Add
trait,它以 String
作为左侧,以 &str
作为右侧:
fn main() {
let owned_string: String = "hello ".to_owned();
let borrowed_string: &str = "world";
let new_owned_string = owned_string + borrowed_string;
println!("{}", new_owned_string);
}
请注意,在调用 +
后,将无法再访问 owned_string
。
如果我们想产生一个新的字符串,而让两者都保持不变呢?最简单的方法是使用 format!
:
fn main() {
let borrowed_string: &str = "hello ";
let another_borrowed_string: &str = "world";
let together = format!("{}{}", borrowed_string, another_borrowed_string);
// After https://rust-lang.github.io/rfcs/2795-format-args-implicit-identifiers.html
// let together = format!("{borrowed_string}{another_borrowed_string}");
println!("{}", together);
}
请注意,两个输入变量都是不可变的,因此我们知道它们没有被触及。如果我们想对 String
的任何组合做同样的事情,我们可以使用 String
也可以格式化的事实:
fn main() {
let owned_string: String = "hello ".to_owned();
let another_owned_string: String = "world".to_owned();
let together = format!("{}{}", owned_string, another_owned_string);
// After https://rust-lang.github.io/rfcs/2795-format-args-implicit-identifiers.html
// let together = format!("{owned_string}{another_owned_string}");
println!("{}", together);
}
不过,您必须使用 format!
。您可以 clone one string 并将另一个字符串附加到新字符串:
fn main() {
let owned_string: String = "hello ".to_owned();
let borrowed_string: &str = "world";
let together = owned_string.clone() + borrowed_string;
println!("{}", together);
}
注意 - 我所做的所有类型规范都是多余的 - 编译器可以在这里推断出所有类型。我添加它们只是为了让刚接触 Rust 的人清楚,因为我希望这个问题会受到该群体的欢迎!
要将多个字符串连接成一个字符串,由另一个字符分隔,有几种方法。
我见过的最好的方法是在数组上使用 join
方法:
fn main() {
let a = "Hello";
let b = "world";
let result = [a, b].join("\n");
print!("{}", result);
}
根据您的用例,您可能还喜欢更多控制:
fn main() {
let a = "Hello";
let b = "world";
let result = format!("{}\n{}", a, b);
print!("{}", result);
}
我见过更多的手动方法,一些避免了这里和那里的一两次分配。出于可读性的目的,我发现以上两个就足够了。
join
记录在哪里?它似乎位于数组和字符串之间。我搜索了 array 文档并很快感到困惑。
join
实际上附加到 the SliceContactExt
trait。该特征被标记为不稳定,但它的方法是稳定的并且 are included in the Prelude 因此默认情况下它们可以在任何地方使用。团队似乎很清楚这个特性不需要存在,我想未来事情会随着它而改变。
join
在连接两个 str
时比 s1.to_owned().push_str(s2)
更有效,因为它避免了第二次分配。
Rust 中连接字符串的简单方法
Rust 中有多种方法可以连接字符串
第一种方法(使用 concat!() ):
fn main() {
println!("{}", concat!("a", "b"))
}
上述代码的输出是:
抗体
第二种方法(使用 push_str() 和 + 运算符):
fn main() {
let mut _a = "a".to_string();
let _b = "b".to_string();
let _c = "c".to_string();
_a.push_str(&_b);
println!("{}", _a);
println!("{}", _a + &_c);
}
上述代码的输出是:
美国广播公司
第三种方法(使用格式!()):
fn main() {
let mut _a = "a".to_string();
let _b = "b".to_string();
let _c = format!("{}{}", _a, _b);
println!("{}", _c);
}
上述代码的输出是:
抗体
检查它并尝试使用 Rust playground。
concat
仅适用于文字,因此不是超级有用。
我认为这里也应该提到 concat
方法和 +
:
assert_eq!(
("My".to_owned() + " " + "string"),
["My", " ", "string"].concat()
);
还有 concat!
宏,但仅适用于文字:
let s = concat!("test", 10, 'b', true);
assert_eq!(s, "test10btrue");
通过字符串插值连接
更新:截至 2021 年 12 月 28 日,这在 Rust 1.58 Beta 中可用。您不再需要 Rust Nightly 构建来执行字符串插值。 (为后代保留答案的其余部分不变)。
RFC 2795 于 2019 年 10 月 27 日发布:建议支持隐式参数来执行许多人所知的“字符串插值”——一种在字符串中嵌入参数以连接它们的方法。
RFC:https://rust-lang.github.io/rfcs/2795-format-args-implicit-identifiers.html
可在此处找到最新问题状态:https://github.com/rust-lang/rust/issues/67984
在撰写本文时(2020-9-24),我相信这个功能应该在 Rust Nightly 版本中可用。
这将允许您通过以下速记进行连接:
format_args!("hello {person}")
它等价于:
format_args!("hello {person}", person=person)
还有“ifmt”板条箱,它提供了自己的字符串插值:
从 Rust 1.58 开始,您还可以像这样连接两个或多个变量:format!("{a}{b}{c}")
。这与 format!("{}{}{}", a, b, c)
基本相同,但更短一些并且(可以说)更易于阅读。这些变量可以是 String
、&str
(也可以是其他非字符串类型)。结果是 String
。有关更多信息,请参见this。
默认情况下,Rust 中的所有内容都是关于 MemoryManage 和 Owenership 和 Move,我们通常看不到复制或深复制,因此如果您尝试连接字符串,那么左侧应该输入可增长的字符串,并且应该是可变类型,右侧可以是普通的字符串文字,也就是类型字符串切片
fn main (){
let mut x = String::from("Hello"); // type String
let y = "World" // type &str
println!("data printing -------> {}",x+y);
}
https://i.stack.imgur.com/h6WQI.png
fn main() {
let a = String::from("Name");
let b = "Pkgamer";
println!("{}",a+b)
}
Add
/+
符号?如果你愿意,你可以覆盖它。.to_owned()
和.to_string()
已得到修复。现在,它们在&str
上调用时具有相同的性能。相关提交:github.com/rust-lang/rust/pull/32586/filesString
,但随后您引用了一个 (&String
),它可以强制转换为&str
。我把整个路径String
->&String
->&str
,因为初学者甚至可能没有意识到您可以参考String
。 :-)