Выбрать главу

XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX

Параметры фантомного типа - единственное, что не отображается во время выполнения, но проверяется статически (и только статически) во время компиляции.

Типы данных могут использовать дополнительные обобщённые типы в качестве параметров-маркеров или для выполнения проверки типов во время компиляции. Эти дополнительные параметры не сохраняют значения и не имеют поведения во время выполнения.

В следующем примере мы совместили std::marker::PhantomData и концепцию параметров фантомных типов для создания кортежей разных типов.

use std::marker::PhantomData;

// Фантомная кортежная структура, которая имеет обобщение `A` со скрытым параметром `B`.

#[derive(PartialEq)] // Разрешаем для данного типа сравнения.

struct PhantomTuple<A, B>(A,PhantomData<B>);

// Фантомная структура, которая имеет обобщение `A` со скрытым параметром `B`.

#[derive(PartialEq)] // Разрешаем для данного типа сравнения.

struct PhantomStruct<A, B> { first: A, phantom: PhantomData<B> }

// Заметьте: память выделена для обобщённого типа `A`, но не для `B`.

// Следовательно, `B` не может быть использована в вычислениях.

fn main() {

// Здесь `f32` и `f64` - скрытые параметры.

// Тип PhantomTuple объявлен с `<char, f32>`.

let _tuple1: PhantomTuple<char, f32> = PhantomTuple('Q', PhantomData);

// Тип PhantomTuple объявлен с `<char, f64>`.

let _tuple2: PhantomTuple<char, f64> = PhantomTuple('Q', PhantomData);

// Тип определён как `<char, f32>`.

let _struct1: PhantomStruct<char, f32> = PhantomStruct {

first: 'Q',

phantom: PhantomData,

};

// Тип определён как `<char, f64>`.

let _struct2: PhantomStruct<char, f64> = PhantomStruct {

first: 'Q',

phantom: PhantomData,

};

// Ошибка времени компиляции! Типы не совпадают, так что сравнение не может быть произведено:

//println!("_tuple1 == _tuple2 даёт в результате: {}",

// _tuple1 == _tuple2);

// Ошибка времени компиляции! Типы не совпадают, так что сравнение не может быть произведено:

//println!("_struct1 == _struct2 даёт в результате: {}",

// _struct1 == _struct2);

}

הההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההההה

XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX

derive, struct и кортежные структуры

Полезный метод преобразования единиц измерения может быть получен путём реализации типажа Add с параметром фантомного типа. trait``Add рассмотрен ниже:

// Эта конструкция будет навязывать: `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;

/// Создаём пустые перечисления для определения типов единиц измерения.

#[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` объявляет поведение оператора `+`.

impl<Unit> Add for Length<Unit> {

type Output = Length<Unit>;

// add() возвращает новую структуру `Length`, содержащую сумму.

fn add(self, rhs: Length<Unit>) -> Length<Unit> {

// `+` вызывает реализацию `Add` для `f64`.

Length(self.0 + rhs.0, PhantomData)

}

}

fn main() {

// Объявим, что `one_foot` имеет парамет фантомного типа `Inch`.