12.8.1 Testcase: unit clarification

단위 변환에 유용한 방법은 유령 타입 매개자와 Add의 구현으로 검사할 수 있다. Add``trait은 아래처럼 시험된다:

// 이 생산자가 부여하는 것: `Self + RHS = Output`
// 구현 시에 지정되지 않으면 RHS의 기본 값은 Self.
pub trait Add<RHS = Self> {
    type Output;

    fn add(self, rhs: RHS) -> Self::Output;
}

// `Output`은 반드시 `T<U>`여야 한다 그래서 `T<U> + T<U> = T<U>`.
impl<U> Add for T<U> {
    type Output = T<U>;
    ...
}

전체 구현이다 :

use std::ops::Add;
use std::marker::PhantomData;

/// 단위 타입을 정의하는 void 열거형을 만든다.
#[derive(Debug, Clone, Copy)]
enum Inch {}
#[derive(Debug, Clone, Copy)]
enum Mm {}

///`Length`는 유령 타입 매개 변수 `Unit`를 가진 타입이고, 
// 길이 타입 (이는 `f64`)에 대해서는 제네릭하지 않다. 
// `f64`는 이미 `Clone`과 `Copy` 특성을 구현한다.
#[derive(Debug, Clone, Copy)]
struct Length<Unit>(f64, PhantomData<Unit>);

/// `Add` trait은 `+`연산자의 행위를 정의한다.
impl<Unit> Add for Length<Unit> {
     type Output = Length<Unit>;

    // add()는 합계를 담고있는 새로운 `Length` 구조체를 반환한다.
    fn add(self, rhs: Length<Unit>) -> Length<Unit> {
        // `+` calls the `Add` implementation for `f64`.
        Length(self.0 + rhs.0, PhantomData)
    }
}

fn main() {
    // 유령 타입 매개변수 `Inch`를 갖도록 `one_foot`를 지정
    let one_foot:  Length<Inch> = Length(12.0, PhantomData);
    // `one_meter`는 유령 타입 매개변수 `Mm`을 갖는다.
    let one_meter: Length<Mm>   = Length(1000.0, PhantomData);

    // `+`는 `Length`에 구현한 `add()`메소드를 호출한다. 
    // `Length`는 `Copy`를 구현하여, `add()`는 `one_foot`와 `one_meter`를 
    // 소비하지 않고 `self`와 `rhs`로 복사한다.
    let two_feet = one_foot + one_foot;
    let two_meters = one_meter + one_meter;

    // 추가 작업.
    println!("one foot + one_foot = {:?} in", two_feet.0);
    println!("one meter + one_meter = {:?} mm", two_meters.0);

    // 다음과 같이 비정상적 동작 실패를 한다:
    // Compile-time Error: type mismatch.
    //let one_feter = one_foot + one_meter;
}

See also:

Borrowing (&), Bounds (X: Y), enum, impl & self, Overloading, ref, Traits (X for Y), and TupleStructs.

results matching ""

    No results matching ""