std :: make_tuple не делает ссылки - PullRequest
       12

std :: make_tuple не делает ссылки

35 голосов
/ 23 октября 2011

Я экспериментировал с std::tuple в сочетании со ссылками:

#include <iostream>
#include <tuple>

int main() {
  int a,b;
  std::tuple<int&,int&> test(a,b);
  std::get<0>(test) = 1;
  std::get<1>(test) = 2;
  std::cout << a << ":" << b << std::endl;

  // doesn't make ref, not expected
  auto test2 = std::make_tuple(a,b);
  std::get<0>(test2) = -1;
  std::get<1>(test2) = -2;
  std::cout << a << ":" << b << std::endl;

  int &ar=a;
  int &br=b;
  // why does this not make a tuple of int& references? can we force it to notice?
  auto test3 = std::make_tuple(ar,br);
  std::get<0>(test3) = -1;
  std::get<1>(test3) = -2;
  std::cout << a << ":" << b << std::endl;
}

Из трех примеров здесь первые два работают, как и ожидалось. Третий, однако, нет. Я ожидал, что тип auto (test3) будет таким же, как тип test (т.е. std::tuple<int&,int&>).

Кажется, что std::make_tuple не может автоматически создавать кортежи ссылок. Почему бы и нет? Что я могу сделать, чтобы это так, кроме явного создания чего-то такого же типа?

(компилятор был g ++ 4.4.5, при использовании 4.5 не меняет его )

Ответы [ 6 ]

53 голосов
/ 23 октября 2011

std::tie делает не-const ссылки.

auto ref_tuple = std::tie(a,b); // decltype(ref_tuple) == std::tuple<int&, int&>

Для const ссылок вам может понадобиться функция оболочки std::cref:

auto cref_tuple = std::make_tuple(std::cref(a), std::cref(b));

Или используйте простой помощник as_const, чтобы квалифицировать переменные перед тем, как передать их в std::tie:

template<class T>
T const& as_const(T& v){ return v; }

auto cref_tuple = std::tie(as_const(a), as_const(b));

Или, если вы хотите проявить фантазию, напишите свой собственный ctie (повторное использование std::tieи as_const):

template<class... Ts>
std::tuple<Ts const&...> ctie(Ts&... vs){
  return std::tie(as_const(vs)...);
}

auto cref_tuple = ctie(a, b);
29 голосов
/ 23 октября 2011

Попробуйте forward_as_tuple:

auto test3 = std::forward_as_tuple(ar,br);
4 голосов
/ 23 октября 2011

Как насчет:

auto test3 = std::make_tuple(std::ref(a),std::ref(b));
3 голосов
/ 23 октября 2011

Почему: make_tuple параметры передаются по константной ссылке (const T&), поэтому, если вы передаете int&, T соответствует int.Если вывести T в int&, параметр будет const T&&, и вы получите ошибку компиляции.

0 голосов
/ 06 апреля 2017

В C ++ 14 вы можете действовать следующим образом:

template<typename ...T, size_t... I>
auto make_rtuple_helper(std::tuple<T...>& t ,  std::index_sequence<I...>)
-> std::tuple<T&...>
{ return std::tie(std::get<I>(t)...) ;}

template<typename ...T>
std::tuple<T&...> make_rtuple( std::tuple<T...>& t )
{
    return make_rtuple_helper( t, std::make_index_sequence<sizeof...(T)>{});
}

Посмотрите, как это работает здесь, в coliru: http://coliru.stacked -crooked.com / a / a665130e17fd8bcc

Приветствие * 1009 A.A. *

0 голосов
/ 23 октября 2011

Как насчет std::make_tuple<int&, int&>(a, b);

По общему признанию, это отчасти побеждает цель, но для таких функций, как make_shared, вы все равно получаете преимущества.

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

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...