Vec<T>,整数向量 v​​s 字符串向量,为什么我可以 index/copy int 元素,即使元素在堆上?

Vec<T>, vector of intergers vs vector of String, why can I index/copy int element even if the elements are on the heap?

简而言之,一些数据类型存储在堆栈中,因为编译器知道它们在 运行 时需要多少内存。其他数据类型更灵活,存储在堆中。数据的Pointer留在栈上,指向堆数据。

我的问题是,如果 Vec 数据在堆上,那么如何访问 i32(和其他通常堆栈存储的类型)就好像它们实际上在堆栈上(通过索引复制)一样。

也就是说。我不能 move 从 Vec 中取出字符串,这对我来说很有意义,它们不实现复制,通常是 move。当它们是 Vec 的元素时,也会发生同样的情况。然而,i32 通常被复制,但为什么当它们是堆上向量数据的一部分时也会发生这种情况?

请随时指出任何概念上的错误,如果您认为我遗漏了什么,请指出现有的 material。我已经阅读了 Rust 编程语言并进行了一些检查。

fn main() {
    // int in stack
    let i: i32 = 1;
    let _ic = i;
    println!("{}", i);

    // String on heap
    let s: String = String::from("ciao cippina");
    let _sc = &s;
    println!("{}", s);

    // array and data on the stack
    let ari = [1, 2, 3];
    println!("{:?}", &ari);
    println!("a 0 {}", ari[0]);

    // array and Pointers on the stack, data on the heap
    let ars = [String::from("ciao"), String::from("mondo")];
    println!("{:?}", &ars);
    println!("a 0 {}", ars[0]);
    // let _ars_1 = ars[0];  // ERROR, cannot move out of array

    // Vec int, its Pointer on stack, all the rest on heap
    let veci = vec![2, 4, 5, 6];
    println!("{:?}", &veci);
    println!("a 0 {}", veci[0]);
    let _veci_1 = veci[0];  // NO ERROR HERE ??

    // Vec string, its Pointer on stack, all the rest on heap
    let vecs = vec![String::from("ciao"), String::from("mondo")];
    println!("{:?}", &vecs);
    println!("a 0 {}", vecs[0]);
    // let _vecs_1 = vecs[0];  // ERROR, cannot move out of Vec
}

仅仅因为向量的元素存在于堆上并不意味着编译器无法知道元素的大小。元素存在于何处并不重要,如果类型是 "copyable",则可以从堆栈复制 -> 堆,反之亦然。

在你的例子中,i32 占用 4 个字节,无论是在堆上还是在堆栈上(忽略对齐问题)