배열과 슬라이스

Beginner

This tutorial is from open-source community. Access the source code

소개

이 랩에서는 Rust 의 배열 (array) 과 슬라이스 (slice) 를 탐구합니다. 배열은 동일한 유형의 객체들을 연속된 메모리에 저장하는 컬렉션이며, 그 길이는 컴파일 시간에 알려져 있습니다. 반면에 슬라이스는 배열과 유사하지만, 그 길이는 컴파일 시간에 알려져 있지 않습니다. 슬라이스는 배열의 일부분을 빌려오는 데 사용될 수 있습니다. 또한 배열을 생성하고, 요소에 접근하며, 길이를 계산하고, 메모리를 할당하고, 배열을 슬라이스로 빌려오고, 빈 슬라이스를 사용하는 방법도 다룰 것입니다. 더불어, .get() 메서드를 사용하여 배열 요소에 안전하게 접근하고, 범위를 벗어난 (out-of-bounds) 오류를 처리하는 방법도 논의할 것입니다.

참고: 랩에서 파일 이름을 지정하지 않은 경우, 원하는 파일 이름을 사용할 수 있습니다. 예를 들어, main.rs를 사용하고, rustc main.rs && ./main으로 컴파일하고 실행할 수 있습니다.

배열과 슬라이스

배열은 동일한 타입 T의 객체들을 연속된 메모리에 저장하는 컬렉션입니다. 배열은 대괄호 []를 사용하여 생성되며, 컴파일 시간에 알려진 그 길이는 타입 시그니처 [T; length]의 일부입니다.

슬라이스는 배열과 유사하지만, 그 길이는 컴파일 시간에 알려져 있지 않습니다. 대신, 슬라이스는 두 단어 (two-word) 객체입니다. 첫 번째 단어는 데이터에 대한 포인터이고, 두 번째 단어는 슬라이스의 길이입니다. 단어 크기는 프로세서 아키텍처에 의해 결정되는 usize와 동일합니다. 예를 들어, x86-64 에서는 64 비트입니다. 슬라이스는 배열의 일부분을 빌려오는 데 사용될 수 있으며, 타입 시그니처는 &[T]입니다.

use std::mem;

// This function borrows a slice.
fn analyze_slice(slice: &[i32]) {
    println!("First element of the slice: {}", slice[0]);
    println!("The slice has {} elements", slice.len());
}

fn main() {
    // Fixed-size array (type signature is superfluous).
    let xs: [i32; 5] = [1, 2, 3, 4, 5];

    // All elements can be initialized to the same value.
    let ys: [i32; 500] = [0; 500];

    // Indexing starts at 0.
    println!("First element of the array: {}", xs[0]);
    println!("Second element of the array: {}", xs[1]);

    // `len` returns the count of elements in the array.
    println!("Number of elements in array: {}", xs.len());

    // Arrays are stack allocated.
    println!("Array occupies {} bytes", mem::size_of_val(&xs));

    // Arrays can be automatically borrowed as slices.
    println!("Borrow the whole array as a slice.");
    analyze_slice(&xs);

    // Slices can point to a section of an array.
    // They are of the form [starting_index..ending_index].
    // `starting_index` is the first position in the slice.
    // `ending_index` is one more than the last position in the slice.
    println!("Borrow a section of the array as a slice.");
    analyze_slice(&ys[1 .. 4]);

    // Example of empty slice `&[]`:
    let empty_array: [u32; 0] = [];
    assert_eq!(&empty_array, &[]);
    assert_eq!(&empty_array, &[][..]); // Same but more verbose

    // Arrays can be safely accessed using `.get`, which returns an
    // `Option`. This can be matched as shown below, or used with
    // `.expect()` if you would like the program to exit with a nice
    // message instead of happily continue.
    for i in 0..xs.len() + 1 { // Oops, one element too far!
        match xs.get(i) {
            Some(xval) => println!("{}: {}", i, xval),
            None => println!("Slow down! {} is too far!", i),
        }
    }

    // Out of bound indexing on array causes compile time error.
    //println!("{}", xs[5]);
    // Out of bound indexing on slice causes runtime error.
    //println!("{}", xs[..][5]);
}

요약

축하합니다! 배열과 슬라이스 랩을 완료했습니다. LabEx 에서 더 많은 랩을 연습하여 실력을 향상시킬 수 있습니다.