第8回です。
前回はこちら。
[第8回の様子]
2021/09/01に第8回を開催した。
今回は最初から無事参加できた。
内容としてはRust By Example 日本語版の「2.3. 配列とスライス」を読んで手を動かした。
参加者は全部で9人。ちょっと減ったけど、相変わらずなかなか盛況である。
今回も、前回終了時に決めておいたメンバーがドライバーを担当。
今週も開催前後のSlackでのやりとりがたくさん🎉
遅れても参加してえらい!
楽しく配列のお勉強。
また参加者が増えるかも!
[学んだこと]
- 2.3. 配列とスライス
- 配列は単一の型のオブジェクトの集合
- メモリ上の連続した領域に保存される
- サイズはコンパイル時に決定されて、
[T; size]
型でサイズを指定できる - スライスは配列と似ているが、コンパイル時にはサイズが決定されていない
- しかし後からサイズを変更できるわけではない:その場合は動的に拡張できるVectorを使う
- スライスは配列の一部を借用するときに&[T]という型シグネチャで利用する
- 配列を宣言するときはこんなかんじ
let xs: [i32; 5] = [1, 2, 3, 4, 5]; // 型宣言はなしでもよい let xs = [1, 2, 3, 4, 5];
- 配列を同じ値で初期化することもできる
let ys: [i32; 500] = [0; 500]; // 型宣言はなしでもよい let ys = [0; 500];
- 配列のインデックスは0からはじまる。
let xs = [1, 2, 3, 4, 5]; println!("first element of the array: {}", xs[0]); // 1 println!("second element of the array: {}", xs[1]); // 2
- 配列のサイズを取得するときは
len()
を使う
let xs = [1, 2, 3, 4, 5]; println!("array size: {}", xs.len()); // 5
- あ
use std::mem; let xs = [1, 2, 3, 4, 5]; println!("array occupies {} bytes", mem::size_of_val(&xs)); // 20 bytes // デフォルトはi32で32ビット=4バイト、4x5で20バイト
- 配列を関数に渡すときに、自動的にスライスとして借用される
// この関数はスライスを借用する fn analyze_slice(slice: &[i32]) { println!("first element of the slice: {}", slice[0]); println!("the slice has {} elements", slice.len()); } let xs = [1, 2, 3, 4, 5]; analyze_slice(&xs); // first element of the slice: 1 // the slice has 5 elements
- 配列の一部を借用することもできる
fn analyze_slice(slice: &[i32]) { println!("first element of the slice: {}", slice[0]); println!("the slice has {} elements", slice.len()); } let ys: [i32; 500] = [0; 500]; analyze_slice(&ys[1 .. 4]); // インデックス1から3までの要素をスライスとして借用 // first element of the slice: 0 // the slice has 3 elements
- インデックスが範囲を超えている場合はコンパイルは通るが、実行時にパニックする
let xs = [1, 2, 3, 4, 5]; println!("{}", xs[5]); // 実行時に以下のエラー error: this operation will panic at runtime --> src/main.rs:46:20 | 46 | println!("{}", xs[5]); | ^^^^^ index out of bounds: the length is 5 but the index is 5 | = note: `#[deny(unconditional_panic)]` on by default
- 配列の中身を書き換える場合は、
mut
をつけてlet mut xs = [1, 2]
のように宣言必要がある
let xs: [i32; 5] = [1, 2, 3, 4, 5]; xs[0] = 11; // これはコンパイル時に以下のエラー error[E0594]: cannot assign to `xs[_]`, as `xs` is not declared as mutable --> src/main.rs:14:5 | 13 | let xs: [i32; 5] = [1, 2, 3, 4, 5]; | -- help: consider changing this to be mutable: `mut xs` 14 | xs[0] = 11; | ^^^^^^^^^^ cannot assign
- スライスを可変にするには少々手間がかかる:以下の2つが必要
- 関数の引数として宣言している箇所でmutをつける
- 配列をmutで宣言し、関数を呼び出す際にもmutをつける
fn analyze_slice(slice: &mut[i32]) { slice[0] = 23; println!("first element of the slice: {}", slice[0]); println!("the slice has {} elements", slice.len()); } let mut xs = [1, 2, 3, 4, 5]; analyze_slice(&mut xs); // first element of the slice: 23 // the slice has 5 elements
- スライスを生成する際のインデックス指定も、配列のサイズを超えていると実行時にパニックとなる
- ただし配列へのインデックスアクセスとはエラーメッセージがちょっと異なる
fn analyze_slice(slice: &[i32]) { println!("first element of the slice: {}", slice[0]); println!("the slice has {} elements", slice.len()); } analyze_slice(&ys[1 .. 10]); // thread 'main' panicked at 'range end index 10 out of range for slice of length 5', src/main.rs:42:20 // note: run with `RUST_BACKTRACE=1` environment variable to display a backtrace
[まとめ]
モブプログラミングスタイルでRust dojoを開催した。
配列とスライスで遊んだ。エラーがいろいろ出て楽しい。
今週のプルリクエストはこちら。