Formatação em Rust e a Trait Display

Beginner

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

Introdução

Neste laboratório, aprendemos sobre formatação em Rust e como usar a macro format! para formatar variáveis. Vimos que a formatação é especificada usando uma string de formatação, e diferentes tipos de argumentos podem ser usados para formatar a mesma variável de maneiras diferentes. A trait de formatação mais comum é Display, que lida com casos em que o tipo de argumento não é especificado. Vimos um exemplo de implementação da trait Display para uma struct City, onde formatamos os valores de latitude e longitude. Também vimos um exemplo de uma struct Color e fomos encarregados de implementar a trait Display para ela, a fim de exibir os valores RGB e sua representação hexadecimal.

Nota: Se o laboratório não especificar um nome de arquivo, você pode usar qualquer nome de arquivo que desejar. Por exemplo, você pode usar main.rs, compilar e executá-lo com rustc main.rs && ./main.

Formatação

Vimos que a formatação é especificada por meio de uma format string:

  • format!("{}", foo) -> "3735928559"
  • format!("0x{:X}", foo) -> "0xDEADBEEF"
  • format!("0o{:o}", foo) -> "0o33653337357"

A mesma variável (foo) pode ser formatada de maneira diferente, dependendo do tipo de argumento usado: X vs o vs não especificado.

Esta funcionalidade de formatação é implementada por meio de traits, e há uma trait para cada tipo de argumento. A trait de formatação mais comum é Display, que lida com casos em que o tipo de argumento é deixado não especificado: {} por exemplo.

use std::fmt::{self, Formatter, Display};

struct City {
    name: &'static str,
    // Latitude
    lat: f32,
    // Longitude
    lon: f32,
}

impl Display for City {
    // `f` is a buffer, and this method must write the formatted string into it.
    fn fmt(&self, f: &mut Formatter) -> fmt::Result {
        let lat_c = if self.lat >= 0.0 { 'N' } else { 'S' };
        let lon_c = if self.lon >= 0.0 { 'E' } else { 'W' };

        // `write!` is like `format!`, but it will write the formatted string
        // into a buffer (the first argument).
        write!(f, "{}: {:.3}°{} {:.3}°{}",
               self.name, self.lat.abs(), lat_c, self.lon.abs(), lon_c)
    }
}

#[derive(Debug)]
struct Color {
    red: u8,
    green: u8,
    blue: u8,
}

fn main() {
    for city in [
        City { name: "Dublin", lat: 53.347778, lon: -6.259722 },
        City { name: "Oslo", lat: 59.95, lon: 10.75 },
        City { name: "Vancouver", lat: 49.25, lon: -123.1 },
    ] {
        println!("{}", city);
    }
    for color in [
        Color { red: 128, green: 255, blue: 90 },
        Color { red: 0, green: 3, blue: 254 },
        Color { red: 0, green: 0, blue: 0 },
    ] {
        // Switch this to use {} once you've added an implementation
        // for fmt::Display.
        println!("{:?}", color);
    }
}

Você pode ver uma lista completa de traits de formatação e seus tipos de argumento na documentação std::fmt.

Atividade

Adicione uma implementação da trait fmt::Display para a struct Color acima, para que a saída seja exibida como:

RGB (128, 255, 90) 0x80FF5A
RGB (0, 3, 254) 0x0003FE
RGB (0, 0, 0) 0x000000

Três dicas se você ficar preso:

  • A fórmula para calcular uma cor no espaço de cores RGB é: RGB = (R*65536)+(G*256)+B , (when R is RED, G is GREEN and B is BLUE). Para mais informações, consulte RGB color format & calculation.
  • Você pode precisar listar cada cor mais de uma vez.
  • Você pode preencher com zeros para uma largura de 2 com :0>2.

Resumo

Parabéns! Você concluiu o laboratório de Formatação. Você pode praticar mais laboratórios no LabEx para aprimorar suas habilidades.