У меня есть некоторая функция, которая должна принимать итератор, отдающий

ссылки на некоторый тип, но при этом вложенность ссылок может быть хоть какой, т.е. от &T до, скажем, &&&&&T. Какими трейтами воспользоваться, чтобы описать тип, возвращаемый итератором?

15 ответов

39 просмотров

enum? &T и &&T - это два разных типа, и объединить их в одном типе можно например через enum enum TRef<T> { R1(&T), R2(&&T), R3(&&&T), ... }

Iter: Iterator<Item = Item>, Item: std::ops::Deref<Target=T>,

megahomyak- Автор вопроса
Kirill Kuzminykh
enum? &T и &&T - это два разных типа, и объединить...

Мне не нужно объединять в один тип. Немного больше контекста: fn set_matches_from_actions<Callback, Actions, ActionRef>( matches: &mut Rc<slint::VecModel<ListItem>>, actions: Actions, ) where ActionRef: AsRef<Action<Callback>>, Actions: Iterator<Item = ActionRef>, { matches.set_vec( actions .map(|action| ListItem { text: action.as_ref().name.into(), }) .collect::<Vec<_>>(), ); } Итератор должен иметь возможность возвращать ссылку на Action любой глубины, т.к. поле name нужно лишь иммутабельное. При вызове функции выше как set_matches_from_actions(&mut matches, all_actions.iter());, где all_actions - Vec<Action>, получаю следующее: the trait bound `Action<[closure@src\main.rs:41:9: 43:10]>: std::convert::AsRef<Action<_>>` is not satisfied required because of the requirements on the impl of `std::convert::AsRef<Action<_>>` for `&Action<[closure@src\main.rs:41:9: 43:10]>

megahomyak- Автор вопроса
red75prime
Iter: Iterator<Item = Item>, Item: std::ops::Deref...

Попробовал это, на строчке set_matches_from_actions(&mut matches, matching_actions.iter());, где matching_actions - Vec<&mut Action>, получаю ```type mismatch resolving <&&mut Action<[closure@src\main.rs:41:9: 43:10]> as std::ops::Deref>::Target == Action<_> expected struct Action<_> found mutable reference `&mut Action<[closure@src\main.rs:41:9: 43:10]>````

megahomyak
Мне не нужно объединять в один тип. Немного больше...

Для ссылок разной глубины надо разный код для получения значения по этой ссылке. Просто так не получится сделать общий код, который будет работать с ссылками любой глубины вложенности. Только если в где-то в рантайме будет храниться информация о глубине ссылок и тогда можно будет замутить что-то для deref-а такой ссылки (например match по enum-у, или какую-ту рекурсивную функцию с unsafe-ами).

megahomyak- Автор вопроса
megahomyak
Мне не нужно объединять в один тип. Немного больше...

Ещё больше контекста: Код: https://www.toptal.com/developers/hastebin/isimecerer.rust (на плейграунде всё равно не запустится) Ошибка: https://www.toptal.com/developers/hastebin/oyiyobepid.rust

megahomyak- Автор вопроса
Kirill Kuzminykh
Для ссылок разной глубины надо разный код для полу...

В итераторе ссылки всегда будут одной и той же глубины, я просто не хочу писать несколько функций для обработки ссылок разной вложенности, т.е. тут нужно что-то сделать с дженериками

megahomyak
В итераторе ссылки всегда будут одной и той же глу...

Рекурсивные дженерики? Хм, интересная мысля. Это наверное что бы совсем было больно от работы с дженериками в расте 😊

https://play.rust-lang.org/?version=stable&mode=debug&edition=2021&gist=8cfa24ad35ce68cea4c3992a91000ebb Должно работать

megahomyak- Автор вопроса
red75prime
https://play.rust-lang.org/?version=stable&mode=de...

Хм. Значит, у меня что-то не так. Судя по ошибке от самого rustc (была отправлена выше на hastebin, отправляю картинкой для удобства; до отправки ошибки я смотрел только на выдачу rust-analyzer'a на конкретной строке), происходит что-то не очень тривиальное. Пойду копать, т.к. если у вас работает, то у меня тоже должно заработать, если написать нормально

megahomyak- Автор вопроса
megahomyak
screenshot Хм. Значит, у меня что-то не так. Судя по ошибке о...

Если что, те же самые код и ошибка лежат здесь

red75prime
https://play.rust-lang.org/?version=stable&mode=de...

Может оно работает по той же причине по которой работает вот это? fn main() { let v = 1; println!("{:?}", &v); println!("{:?}", &&v); } Т.е. в твоём примере на самом деле у нас в результате дерефа не получается на выходе &T или T. Хотя видимо я не правильно понял что требуется.

Kirill Kuzminykh
Может оно работает по той же причине по которой ра...

Да, действительно. Там для T выводится не тот тип.

red75prime
Да, действительно. Там для T выводится не тот тип.

У тебя достаточно вот так сделать, и тоже будет работать fn foo<Iter, Item>(it: Iter) where Iter: Iterator<Item = Item>, Item: std::fmt::Debug, { for i in it { println!("{i:?}"); } }

megahomyak
screenshot Хм. Значит, у меня что-то не так. Судя по ошибке о...

Вот, погляди - может тебе так же сделать? https://play.rust-lang.org/?version=stable&mode=debug&edition=2021&gist=b5a857c80518972122cdb37522512072

Похожие вопросы

Обсуждают сегодня

Всем привет. Ребята, подскажите, пожалуйста. у ботов есть ограничение на отправку сообщений - 30 сообщений в секунду, эти ограничения накладываются на все сообщения? или на со...
Artem Stormageddon
4
Блин, ребята, сори за тупые вопросы. А можно ли как-то открыть вебапку по нажатию на кнопку в меню(которое появляется слева, команды)?
Artem Stormageddon
3
Ребята, всем привет. Подскажите, пожалуйста, можно ли как-то через бота понять, что этого бота добавили в группу\канал и выдали ему права администратора?
Artem Stormageddon
9
Привет всем! Почему этот код не срабатывает при добавлении или удалении пользователя из чата? bot.on('chat_member', async (ctx) => { console.log(ctx); }) bot.launch({allo...
Alexander
7
Всем привет. Не понимаю, в чём тут шутка юмора. Убирается только разрешение на send_messages. А send_media_messages остаётся. Как сделать, чтобы оба убирались? await b...
Alexander
2
Есть тут кто занимается разработкой серваков майна? Или знакомые
meow *
3
'frakturBold' => ['𝖆', '𝖇', '𝖈', '𝖉', '𝖊', '𝖋', '𝖌', '𝖍', '𝖎', '𝖏', '𝖐', '𝖑', '𝖒', '𝖓', '𝖔', '𝖕', '𝖖', '𝖗', '𝖘', '𝖙', '𝖚', '𝖛', '𝖜', '𝖝', '𝖞', '𝖟', '𝕬', '𝕭', '𝕮', '𝕯'...
Roma
4
Есть ли лимиты на кол-во вебхук по домену? Стоит в данный момент 900+ ботов и бывает бот перестает отвечать (не приходят вебхуки) 🐒 Помогает только перезапуск
ᅠ [ Кому не ответил, дублируйте ]
11
а что делать если тебя убивают на картах?
Yarik yarik kyda ti lezesh
43
Товарищи, здравствуйте Подскажите, пожалуйста, может кто-нибудь сталкивался с такой задачей Через вебапку можно сканировать qr-код, а есть ли возможность считывать nfc?
Artem Stormageddon
8
Карта сайта