Dia 1 ExercĂcios matinais
Matrizes (Arrays) e Loops (Laços) for
// Copyright 2022 Google LLC // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. // ANCHOR: transpose fn transpose(matrix: [[i32; 3]; 3]) -> [[i32; 3]; 3] { // ANCHOR_END: transpose let mut result = [[0; 3]; 3]; for i in 0..3 { for j in 0..3 { result[j][i] = matrix[i][j]; } } return result; } // ANCHOR: pretty_print fn pretty_print(matrix: &[[i32; 3]; 3]) { // ANCHOR_END: pretty_print for row in matrix { println!("{row:?}"); } } // ANCHOR: tests #[test] fn test_transpose() { let matrix = [ [101, 102, 103], // [201, 202, 203], [301, 302, 303], ]; let transposed = transpose(matrix); assert_eq!( transposed, [ [101, 201, 301], // [102, 202, 302], [103, 203, 303], ] ); } // ANCHOR_END: tests // ANCHOR: main fn main() { let matrix = [ [101, 102, 103], // <-- o comentĂĄrio faz com que o rustfmt adicione uma nova linha [201, 202, 203], [301, 302, 303], ]; println!("matriz:"); pretty_print(&matrix); let transposed = transpose(matrix); println!("transposta:"); pretty_print(&transposed); }
Bonus question
Isso necessita a utilização de conceitos mais avançados. Pode parecer que poderĂamos usar uma slice de slices (&[&[i32]]
) como o tipo de entrada para transposta
e, assim, fazer nossa função lidar com qualquer tamanho de matriz. No entanto, isso falha rapidamente: o tipo de retorno não pode ser &[&[i32]]
, pois ele precisa possuir os dados que vocĂȘ retorna.
VocĂȘ pode tentar usar algo como Vec<Vec<i32>>
, mas isso tambĂ©m nĂŁo funciona muito bem: Ă© difĂcil converter de Vec<Vec<i32>>
para &[&[i32]]
entĂŁo agora vocĂȘ tambĂ©m nĂŁo pode usar impressao_formatada
facilmente.
Assim que chegarmos aos traits and generics, podemos usar o trait std::convert::AsRef
para abstrair qualquer coisa que pode ser referenciada como um slice.
use std::convert::AsRef; use std::fmt::Debug; fn pretty_print<T, Line, Matrix>(matrix: Matrix) where T: Debug, // A line references a slice of items Line: AsRef<[T]>, // A matrix references a slice of lines Matrix: AsRef<[Line]> { for row in matrix.as_ref() { println!("{:?}", row.as_ref()); } } fn main() { // &[&[i32]] pretty_print(&[&[1, 2, 3], &[4, 5, 6], &[7, 8, 9]]); // [[&str; 2]; 2] pretty_print([["a", "b"], ["c", "d"]]); // Vec<Vec<i32>> pretty_print(vec![vec![1, 2], vec![3, 4]]); }
Além disso, o próprio tipo não imporia que as slices filhas tenham o mesmo comprimento, portanto, tal variåvel poderia conter uma matriz invålida.