ChatGPT解决这个技术问题 Extra ChatGPT

如何连接字符串?

如何连接以下类型组合:

str 和 str

字符串和字符串

字符串和字符串

请注意,str&str不同的类型,在 99% 的情况下,您只需关心 &str。还有其他问题详细说明了它们之间的差异。
这回答了你的问题了吗? How to concatenate static strings in Rust

S
Shepmaster

连接字符串时,需要分配内存来存储结果。最容易开始的是 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);
}

在这里,我们有一个可以变异的自有字符串。这是有效的,因为它可能允许我们重用内存分配。 StringString 的情况与 &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 的人清楚,因为我希望这个问题会受到该群体的欢迎!


您如何看待 Add / + 符号?如果你愿意,你可以覆盖它。
也许这很简单,但理解它需要查看 Add with String 的可能类型签名。
@jsalter 这是一个非常独立的主题,因此作为另一个顶级问题可能会很好。我已经更新以链接到适当的文档(至少我能得到的尽可能接近......)
@ChrisMorgan 应该注意的是,由于 impl 专业化,自上述评论以来,差异 .to_owned().to_string() 已得到修复。现在,它们在 &str 上调用时具有相同的性能。相关提交:github.com/rust-lang/rust/pull/32586/files
@paddyg 是的,这有点微妙。起始类型都是 String,但随后您引用了一个 (&String),它可以强制转换为 &str。我把整个路径 String -> &String -> &str,因为初学者甚至可能没有意识到您可以参考 String。 :-)
S
Shepmaster

要将多个字符串连接成一个字符串,由另一个字符分隔,有几种方法。

我见过的最好的方法是在数组上使用 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 文档并很快感到困惑。
@DuaneJ join 实际上附加到 the SliceContactExt trait。该特征被标记为不稳定,但它的方法是稳定的并且 are included in the Prelude 因此默认情况下它们可以在任何地方使用。团队似乎很清楚这个特性不需要存在,我想未来事情会随着它而改变。
也许您应该提到 join 在连接两个 str 时比 s1.to_owned().push_str(s2) 更有效,因为它避免了第二次分配。
M
Matthias Braun

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


此答案不会为现有答案添加任何新内容。
答案的格式很好,这是有目的的。它可能不会添加任何新内容,但我很高兴@ashwin-rajeev 将其归结为。
concat 仅适用于文字,因此不是超级有用。
s
suside

我认为这里也应该提到 concat 方法和 +

assert_eq!(
  ("My".to_owned() + " " + "string"),
  ["My", " ", "string"].concat()
);

还有 concat! 宏,但仅适用于文字:

let s = concat!("test", 10, 'b', true);
assert_eq!(s, "test10btrue");

+ 已在 existing answer 中提及。 (这是一个 Add 特征的实现,它将 String 作为左侧,将 &str 作为右侧:
诚然,现有的答案是如此广泛,但我没有注意到。
迄今为止最好的答案。只需对字符串使用数组方法或 concat 即可。宏只是方便隐藏一些语法,而不是发明复杂的语法,使核心语言变得晦涩难懂。添加 trait 对对象来说可能很好,但至少会让人感到困惑。
J
JamesHoux

通过字符串插值连接

更新:截至 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”板条箱,它提供了自己的字符串插值:

https://crates.io/crates/ifmt


现在这在 Rust 1.58 Beta 中可用。
a
at54321

从 Rust 1.58 开始,您还可以像这样连接两个或多个变量:format!("{a}{b}{c}")。这与 format!("{}{}{}", a, b, c) 基本相同,但更短一些并且(可以说)更易于阅读。这些变量可以是 String&str(也可以是其他非字符串类型)。结果是 String。有关更多信息,请参见this


M
Murugaraju Perumalla

默认情况下,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


P
Pkzoid Token
fn main() {
    let a = String::from("Name");
    let b = "Pkgamer";
    println!("{}",a+b)
}

如果您想知道为什么会被否决:您没有提供任何解释性文字,您的答案没有完全解决问题(&str/String 的三种组合呢?),以及您的答案并没有增加太多,例如 this one