std::get(std::variant)
От: Marty Пират https://www.youtube.com/channel/UChp5PpQ6T4-93HbNF-8vSYg
Дата: 22.10.25 20:20
Оценка:
Здравствуйте!

Я тут начал довольно активно пользоваться std::variant, и часто возникает задача — узнать, что там во варианте лежит. Также возникает задача получить лежащее там значение.

Мне не нравится то, что std::get принимает std::size_t индекс, а std::variant::index() возвращает std::size_t, а std::holds_alternative принимает тип.

Я каждый раз для каждого variant делаю перечисление enum class, если мой variant — MyVariant, то перечисление — MyVariantKind. Имена альтернатив там в одном порядке. Пишу функцию GetKind и вроде норм. Пока было нужно всего несколько раз, и руками было достаточно просто. Но вообще идея мне понравилась, и я подумал, как бы это дело сделать совместимым с std.

Захотелось:
1) поиметь функцию типа std::index (которой на самом деле вроде нет), которая для variant'а вызывает его index() и кастит к моему enum — что тут лучше придумать? Или есть что-то такое, просто я не в курсе?
2) аналог std::get. Кстати, я что-то всегда вызываю std::get, но, по идее, тут должен работать просто get через ADL, и я могу написать свою пачку перегрузок get в том же NS, где и variant, и всё будет работать просто через вызов get, так?
3) holds_alternative, которая принимает variantKind и variant, и возвращает bool — тут по идее, так же как с get
4) get_if — ну это уже на базе остального делается, вроде

Как это всё лучше по красоте сделать?

ЗЫ Думаю запилить какой-нить генератор кода, который бы мне всю красоту генерил по простецкому описанию, и мой Variant, и VariantKind

ЗЗЫ Приглашается в тему Евгений Музыченко с примером своего compile-time скрипта на его гипотетическом диалекте C++, в котором есть МАКРОСЫ. Каким ему видится решение подобной задачи? Хочу почитать код такого решения.

ЗЗЗЫ А как variant может стать value less? И как это можно продетектить? Это из-за исключений может получиться, или есть другие способы?
Маньяк Робокряк колесит по городу
Re: std::get(std::variant)
От: landerhigh Пират  
Дата: 22.10.25 21:02
Оценка: +1
Здравствуйте, Marty, Вы писали:

M>Здравствуйте!


M>Я тут начал довольно активно пользоваться std::variant, и часто возникает задача — узнать, что там во варианте лежит. Также возникает задача получить лежащее там значение.


std::visit решает все озвученные вопросы без необходимости изобретать велосипеды, которые я поскипал.

M>ЗЗЗЫ А как variant может стать value less? И как это можно продетектить? Это из-за исключений может получиться, или есть другие способы?


Ну вроде да.
Re[2]: std::get(std::variant)
От: Marty Пират https://www.youtube.com/channel/UChp5PpQ6T4-93HbNF-8vSYg
Дата: 22.10.25 21:22
Оценка:
Здравствуйте, landerhigh, Вы писали:

M>>Я тут начал довольно активно пользоваться std::variant, и часто возникает задача — узнать, что там во варианте лежит. Также возникает задача получить лежащее там значение.


L>std::visit решает все озвученные вопросы без необходимости изобретать велосипеды, которые я поскипал.


Через жопу и с кучей писанины, да


M>>ЗЗЗЫ А как variant может стать value less? И как это можно продетектить? Это из-за исключений может получиться, или есть другие способы?


L>Ну вроде да.


Это я видел
Маньяк Робокряк колесит по городу
Re: std::get(std::variant)
От: so5team https://stiffstream.com
Дата: 23.10.25 04:00
Оценка: +1
Здравствуйте, Marty, Вы писали:

M>Мне не нравится то, что std::get принимает std::size_t индекс


Вы точно в этом уверены?
https://en.cppreference.com/w/cpp/utility/variant/get.html:

2) Type-based value accessor: If v holds the alternative T, returns a reference to the value stored in v. Otherwise, throws std::bad_variant_access. The call is ill-formed if T is not a unique element of Types....


M>Я каждый раз для каждого variant делаю перечисление enum class, если мой variant — MyVariant, то перечисление — MyVariantKind. Имена альтернатив там в одном порядке. Пишу функцию GetKind и вроде норм. Пока было нужно всего несколько раз, и руками было достаточно просто. Но вообще идея мне понравилась, и я подумал, как бы это дело сделать совместимым с std.


Даже я, будучи паталогическим велосипедостроителем в терминальной стадии, немного в шоке от вашей идеи. Да не, даже много в шоке.
Re[3]: std::get(std::variant)
От: so5team https://stiffstream.com
Дата: 23.10.25 04:00
Оценка: +1
Здравствуйте, Marty, Вы писали:

M>>>Я тут начал довольно активно пользоваться std::variant, и часто возникает задача — узнать, что там во варианте лежит. Также возникает задача получить лежащее там значение.


L>>std::visit решает все озвученные вопросы без необходимости изобретать велосипеды, которые я поскипал.


M>Через жопу и с кучей писанины, да


Может вы просто не умеете его готовить?
Re[2]: std::get(std::variant)
От: Marty Пират https://www.youtube.com/channel/UChp5PpQ6T4-93HbNF-8vSYg
Дата: 23.10.25 06:21
Оценка:
Здравствуйте, so5team, Вы писали:

M>>Мне не нравится то, что std::get принимает std::size_t индекс


S>Вы точно в этом уверены?


Точно:

1) Index-based value accessor: If v.index() == I, returns a reference to the value stored in v. Otherwise, throws std::bad_variant_access. The call is ill-formed if I is not a valid index in the variant.



M>>Я каждый раз для каждого variant делаю перечисление enum class, если мой variant — MyVariant, то перечисление — MyVariantKind. Имена альтернатив там в одном порядке. Пишу функцию GetKind и вроде норм. Пока было нужно всего несколько раз, и руками было достаточно просто. Но вообще идея мне понравилась, и я подумал, как бы это дело сделать совместимым с std.


S>Даже я, будучи паталогическим велосипедостроителем в терминальной стадии, немного в шоке от вашей идеи. Да не, даже много в шоке.


Ну, бывает.
Маньяк Робокряк колесит по городу
Re[4]: std::get(std::variant)
От: Marty Пират https://www.youtube.com/channel/UChp5PpQ6T4-93HbNF-8vSYg
Дата: 23.10.25 06:22
Оценка:
Здравствуйте, so5team, Вы писали:

L>>>std::visit решает все озвученные вопросы без необходимости изобретать велосипеды, которые я поскипал.


M>>Через жопу и с кучей писанины, да


S>Может вы просто не умеете его готовить?


Возможно. Научи, как правильно
Маньяк Робокряк колесит по городу
Re[3]: std::get(std::variant)
От: so5team https://stiffstream.com
Дата: 23.10.25 07:24
Оценка:
Здравствуйте, Marty, Вы писали:

M>>>Мне не нравится то, что std::get принимает std::size_t индекс


S>>Вы точно в этом уверены?


M>Точно:


Вы специально проигнорировали цитату про второй вариант std::get-а?
Re[5]: std::get(std::variant)
От: so5team https://stiffstream.com
Дата: 23.10.25 07:29
Оценка: +5
Здравствуйте, Marty, Вы писали:

S>>Может вы просто не умеете его готовить?


M>Возможно. Научи, как правильно


Может вы покажите как у вас получается через жопу и с большим количеством писанины, чтобы читатели форума могли подсказать как улучшить ситуацию? Или признать вашу правоту.

А то без кода обсуждать можно разве что ваши дурные привычки к функциям на много сотен строк.
Re[3]: std::get(std::variant)
От: landerhigh Пират  
Дата: 23.10.25 09:32
Оценка:
Здравствуйте, Marty, Вы писали:

L>>std::visit решает все озвученные вопросы без необходимости изобретать велосипеды, которые я поскипал.

M>Через жопу и с кучей писанины, да

Можно и через жопу. И даже с кучей писанины. В плюсах вообще можно всё.
Но зачем?
Re[3]: std::get(std::variant)
От: rg45 СССР  
Дата: 23.10.25 18:39
Оценка:
Здравствуйте, Marty, Вы писали:

L>>std::visit решает все озвученные вопросы без необходимости изобретать велосипеды, которые я поскипал.


M>Через жопу и с кучей писанины, да


Да прям уж куча:

https://coliru.stacked-crooked.com/a/3bf34d7526b1245c

    std::visit([](auto&& t) {
        std::cout << "Тут лежит " << t << std::endl;
    }, v);


А как бы ты хотел чтоб это выглядело? Можешь примерчик набросать?
--
Справедливость выше закона. А человечность выше справедливости.
Отредактировано 23.10.2025 18:50 rg45 . Предыдущая версия . Еще …
Отредактировано 23.10.2025 18:42 rg45 . Предыдущая версия .
Отредактировано 23.10.2025 18:40 rg45 . Предыдущая версия .
Re[4]: std::get(std::variant)
От: Marty Пират https://www.youtube.com/channel/UChp5PpQ6T4-93HbNF-8vSYg
Дата: 23.10.25 19:11
Оценка:
Здравствуйте, rg45, Вы писали:

R>Да прям уж куча:


Именно что куча


R>https://coliru.stacked-crooked.com/a/3bf34d7526b1245c


R>
R>    std::visit([](auto&& t) {
R>        std::cout << "Тут лежит " << t << std::endl;
R>    }, v);
R>


R>А как бы ты хотел чтоб это выглядело? Можешь примерчик набросать?


Как-то так:
int func(MyVariant v)
{
    auto kind = getKind(v);
    if (kind==Kind::A)
    {
        doSomething(get<Kind::A>(v));
        // doSomething(get(kind, v)); // или даже так
    }
    else if (kind==Kind::B || kind==Kind::C)
    {
        log << "Error: " << enum_serialize(kind) << " not allowed here\n";
        return -1; // error - B or C not allowed here
    }
    else
    {
        return 0; // OK
    }

    //...

}


Просто и понятно.

Или switch(kind) сделать, или ещё что
Маньяк Робокряк колесит по городу
Отредактировано 23.10.2025 20:00 Marty . Предыдущая версия . Еще …
Отредактировано 23.10.2025 19:18 Marty . Предыдущая версия .
Re[4]: std::get(std::variant)
От: Marty Пират https://www.youtube.com/channel/UChp5PpQ6T4-93HbNF-8vSYg
Дата: 23.10.25 19:19
Оценка:
Здравствуйте, so5team, Вы писали:

S>Вы специально проигнорировали цитату про второй вариант std::get-а?


Да
Маньяк Робокряк колесит по городу
Re: std::get(std::variant)
От: rg45 СССР  
Дата: 23.10.25 19:20
Оценка: +1
Здравствуйте, Marty, Вы писали:

M>Захотелось:

M>1) поиметь функцию типа std::index (которой на самом деле вроде нет), которая для variant'а вызывает его index() и кастит к моему enum — что тут лучше придумать? Или есть что-то такое, просто я не в курсе?
M>2) аналог std::get. Кстати, я что-то всегда вызываю std::get, но, по идее, тут должен работать просто get через ADL, и я могу написать свою пачку перегрузок get в том же NS, где и variant, и всё будет работать просто через вызов get, так?
M>3) holds_alternative, которая принимает variantKind и variant, и возвращает bool — тут по идее, так же как с get
M>4) get_if — ну это уже на базе остального делается, вроде

По-моему, во всех этих вариантах писанины будет существенно больше, чем при использовании std::visit. Для каждого типа варианта нужно написать энум, потом скастить к этому энуму индекс, потом этот энум запихнуть в какой-то switch и в конечном итоге, скорее всего, вызвать для каждого случая какую-то функцию. Ты же не будешь писать пласты кода прямо внутри switch-а.

А через std::get твоя задача (УЗНАТЬ, что лежит в вариарианте) вообще не решается. Рассчитано на то, что ты сам знаешь, что лежит в варианте и вызываешь std::get с правильным индексом или типом, в противном случае получишь исключение.

Третий и четвёртый варианты — это проверить, а не узнать.

В общем, осваивай std::variant, и твоя жизнь сразу упростится.
--
Справедливость выше закона. А человечность выше справедливости.
Re[5]: std::get(std::variant)
От: landerhigh Пират  
Дата: 23.10.25 19:23
Оценка:
Здравствуйте, Marty, Вы писали:

M>Как-то так:

  Скрытый текст
M>
M>int func(MyVariant v)
M>{
M>    auto kind = getKind(v);
M>    if (kind==Kind::A)
M>    {
M>        doSomething(get<Kind::A>(v));
M>        // doSomething(get(kind, v)); // или даже так
M>    }
M>    else if (kind==Kind::B || kind==Kind::C)
M>    {
M>        log << "Error: " << enum_serialize(kind) << " not allowed here\n";
M>        return -1; // error - B or C not allowed here
M>    }
M>    else
M>    {
M>        return 0; // OK
M>    }
M>}
M>


    std::visit([](auto&& t) {
        if constexpr(std::is_same_v(t, Kind::A)) 
        {
        }
        else if constexpr(std::is_same_v(t, Kind::B))
        {
        }
    }, v);




Можно еще и приукрасить, как тут написано.
Re[2]: std::get(std::variant)
От: Marty Пират https://www.youtube.com/channel/UChp5PpQ6T4-93HbNF-8vSYg
Дата: 23.10.25 19:27
Оценка:
Здравствуйте, rg45, Вы писали:

R>По-моему, во всех этих вариантах писанины будет существенно больше, чем при использовании std::visit. Для каждого типа варианта нужно написать энум, потом скастить к этому энуму индекс,


Генератор кода


R>потом этот энум запихнуть в какой-то switch и в конечном итоге, скорее всего, вызвать для каждого случая какую-то функцию. Ты же не будешь писать пласты кода прямо внутри switch-а.


Или не вызвать, а выйти из текущей


R>А через std::get твоя задача (УЗНАТЬ, что лежит в вариарианте) вообще не решается. Рассчитано на то, что ты сам знаешь, что лежит в варианте и вызываешь std::get с правильным индексом или типом, в противном случае получишь исключение.


ну да, для узнать это get_if или holds_alternative


R>Третий и четвёртый варианты — это проверить, а не узнать.


R>В общем, осваивай std::variant, и твоя жизнь сразу упростится.


Я как-то осваивал std::variant, пиша (пися? писая?) тулзу с использованием LLVM. Мягко говоря, осталось впечатление, что это полный п надо бы как-то попроще всё это
Маньяк Робокряк колесит по городу
Re[6]: std::get(std::variant)
От: Marty Пират https://www.youtube.com/channel/UChp5PpQ6T4-93HbNF-8vSYg
Дата: 23.10.25 19:31
Оценка:
Здравствуйте, landerhigh, Вы писали:

Вместо
if (kind==Kind::B || kind==Kind::C)
{
}


у нас намечается простыня


L>
L>    std::visit([](auto&& t) {
L>        if constexpr(std::is_same_v(t, Kind::A)) 
L>        {
L>        }
L>        else if constexpr(std::is_same_v(t, Kind::B))
L>        {
L>        }
L>    }, v);
L>


L>


А как мне выйти из функции, которая вызывает std::visit, по какой-нибудь альтернативе?

Ещё бесит, что в visit значение варианта передаётся после обработчика. Почему нельзя было его первым аргументом? Оно за портянкой лямбды теряется совсем
Маньяк Робокряк колесит по городу
Re[2]: std::get(std::variant)
От: Marty Пират https://www.youtube.com/channel/UChp5PpQ6T4-93HbNF-8vSYg
Дата: 23.10.25 19:40
Оценка:
Здравствуйте, rg45, Вы писали:

R>В общем, осваивай std::variant, и твоя жизнь сразу упростится.


Или вот вот такой пример. Есть variant с парой десятков альтернатив. У половины альтернатив есть атрибут name, у других нет. Я хочу функцию, которая возвращает name, если он есть, или пустую строку, если его нет. Чтобы выше не парится, выписывая везде std::visit для получения этого атрибута.

auto kind = getKind(v);
name = getName(kind, v);


Как мне такое сделать?

Я пока портяночно написал, но вот думаю, как бы можно было бы это лучше сделать.

например:

string getNameImpl(Kind kind, Variant v, initializer_list<Kind> hasNameList)
{
    for(auto k: hasNameList)
    {
        if (k==kind)
            return visit([](auto a) { return a.name; });
    }

    return string();
}

string getName(Kind kind, Variant v)
{
    return getNameImpl(kind, x, {Kind::A, Kind::B, ...});
}


Ну или как-то так

Тут наверное без вариадик тайплиста не обойтись, но хочется как-то попроще
Маньяк Робокряк колесит по городу
Отредактировано 23.10.2025 19:52 Marty . Предыдущая версия . Еще …
Отредактировано 23.10.2025 19:49 Marty . Предыдущая версия .
Re[5]: std::get(std::variant)
От: rg45 СССР  
Дата: 23.10.25 19:52
Оценка: +1
Здравствуйте, Marty, Вы писали:

R>>А как бы ты хотел чтоб это выглядело? Можешь примерчик набросать?


M>Как-то так:

M>
M>int func(MyVariant v)
M>{
M>    auto kind = getKind(v);
M>    if (kind==Kind::A)
M>    {
M>        doSomething(get<Kind::A>(v));
M>        // doSomething(get(kind, v)); // или даже так
M>    }
M>    else if (kind==Kind::B || kind==Kind::C)
M>    {
M>        log << "Error: " << enum_serialize(kind) << " not allowed here\n";
M>        return -1; // error - B or C not allowed here
M>    }
M>    else
M>    {
M>        return 0; // OK
M>    }
M>}
M>


M>Просто и понятно.


Ну и чем это проще и понятнее, чем:

https://coliru.stacked-crooked.com/a/83124dc3a3facc5d

int func(MyVariant v)
{
    return std::visit([]<typename T>(const T& t) {
        if constexpr (std::same_as<T, A>)
        {
            doSomething(t);
        }
        else if constexpr (std::same_as<T, B> or std::same_as<T, C>)
        {
            std::cout << "Error: B or C not allowed here\n";
            return -1;
        }
        return 0; // OK
    }, v);
}


И это не единственный, и не лучший, кстати, вариант использования — тут на перегрузках можно реально по красоте всё сделать.

И я привёл законченный работающий пример, а тебе ещё нужно наколбасить энумов к своим вариантам. В итоге нифига не проще будет, а будет реально куча.
--
Справедливость выше закона. А человечность выше справедливости.
Отредактировано 23.10.2025 19:53 rg45 . Предыдущая версия .
Re[7]: std::get(std::variant)
От: rg45 СССР  
Дата: 23.10.25 19:55
Оценка: +3
Здравствуйте, Marty, Вы писали:


M>у нас намечается простыня


Так это только благодаря тебе она начинается, потому что ты if-ы используешь. А так-то вообще такие штуки на перегрузках делаются. И перегрузки можно делать по-разному. Можно, лямбды использовать, можно собственные функционалы. Можно через композицию, можно через монолиты. И здесь очень востребованы оказываются концепцы и констрейнты. И код реально выглядит красиво и профессионально, не что что твои if-ы, с энумами.
--
Справедливость выше закона. А человечность выше справедливости.
Отредактировано 23.10.2025 20:00 rg45 . Предыдущая версия . Еще …
Отредактировано 23.10.2025 19:59 rg45 . Предыдущая версия .
Подождите ...
Wait...
Пока на собственное сообщение не было ответов, его можно удалить.