Формулируется достаточно просто, но как красиво решить пока не понимаю.
use std::marker::PhantomData;
// Некоторое хранилище с определённым типом.
// Читать из него не нужно, только писать.
// Должен быть параметризован, чтобы разрешать запись значений только одного типа.
struct Storage<T> {
_marker: PhantomData<T>,
}
trait Value {}
impl<T: Value> Storage<T> {
fn new() -> Self { Self { _marker: PhantomData } }
// Метод должен принимать значения с более узким лайфтаймом, чем T,
// То есть параметр должен быть контравариантным по лайфтайму.
fn add(&mut self, _value: &T) {}
}
fn main() {
struct SomeConcreteValue<'a>(&'a str);
impl Value for SomeConcreteValue<'_> {} // генерируется в derive макросе
let mut test: Storage<SomeConcreteValue<'static>> = Storage::new();
let s = "some".to_string();
test.add(&SomeConcreteValue(&s));
}
Поиграться можно тут
https://play.rust-lang.org/?version=stable&mode=debug&edition=2021&gist=68127cd2bba6276223a81b375164e434
Пока лучшее, что придумал, это в derive генерировать некоторый Token и дальше
fn add<U>(&mut self, _value: &U) where U: Value<Token = T::Token> {}
Но может будут идеи лучше?
Вариант с токеном тоже плохой, ибо с дженериками не дружит нормально. Ещё есть вариант фейла в рантайме, если typeid другой, но это рантайм =/
Обсуждают сегодня