tuple
в boost и TR1 / c ++ 0x предоставляет удобный (для автора функции) метод для возврата двух значений из функции - однако, похоже, он повреждает одну важную функцию языка для вызывающей стороны : возможность просто использовать функцию для инициализации переменной:
T happy();
const auto meaningful_name(happy()); // RVO means no excess copies
но для:
tuple<T,U> sad();
мы должны либо отказаться от возможности выбрать значимое имя для наших возвращаемых значений, и использовать get<n>()
везде:
const auto two_unrelated_things(sad());
или сделать временный:
const auto unwanted_named_temporary(sad());
const auto one_name(get<0>(unwanted_named_temporary));
const auto two_name(get<1>(unwanted_named_temporary));
или переключение с инициализации на присвоение, которое работает только тогда, когда типы могут быть назначены, и прерывается auto
:
tuple_element<0, decltype(sad())>::type one_mutable; // there might be a less
tuple_element<1, decltype(sad())>::type two_mutable; // verbose way
tie(one_mutable,two_mutable) = sad();
или сделать что-то неестественное для местного класса:
const struct ugh {
ugh( decltype(sad()) rhs ) : one_name(get<0>(rhs)), two_name(get<1>(rhs)) {}
const tuple_element<0, decltype(sad())>::type one_name;
const tuple_element<1, decltype(sad())>::type two_name;
} stuff(sad()); // at least we avoid the temporary and get initialization
Есть ли лучший способ? Я использую VC10-совместимые конструкции выше, поможет ли что-нибудь в полной версии c ++ 0x или boost?
В идеале это будет:
- позвольте мне использовать инициализацию, а не просто присваивание
- пусть вызывающий выберет имена для возвращаемых переменных
- не делать лишних копий
- работает как для переменных стека, так и для членов класса
- возможно, это большая сумасшедшая библиотека шаблонов, но со здравым синтаксисом для вызывающей и пишущей функции