boost:: преобразуется с помощью функтора кортежей, выдает странное предупреждение
Я написал удобную оболочку функтора для кортежа std:: get. При использовании его с преобразованием boost и operator[] я получаю предупреждение, что возвращаю ссылку на локальный временный объект. Моя система: Ubuntu 14.04, компиляторы: clang-3.5 и g++-4.8.2, буст-версия: 1.56.
#include <boost/range/adaptor/transformed.hpp>
#include <utility>
#include <vector>
template <std::size_t I>
struct tuple_get {
template <typename Tuple>
auto operator()(Tuple &&tuple) const ->
decltype(std::get<I>(std::forward<Tuple>(tuple))) {
return std::get<I>(std::forward<Tuple>(tuple));
}
};
int main() {
//removing const gets rid of warning
const std::vector<std::tuple<int,int>> v = {std::make_tuple(0, 0)};
//gives warning
(v | boost::adaptors::transformed(tuple_get<0>{})) [0];
}
Предупреждение детали:
include/boost/range/iterator_range_core.hpp:390:16: warning: returning reference to local temporary object [-Wreturn-stack-address]
return this->m_Begin[at];
note: in instantiation of member function 'boost::iterator_range_detail::iterator_range_base<boost::transform_iterator<tuple_get<0>,
std::__1::__wrap_iter<const std::__1::tuple<int, int> *>, boost::use_default, boost::use_default>, boost::random_access_traversal_tag>::operator[]' requested here
(v | boost::adaptors::transformed(tuple_get<0>{})) [0];
Добавление флага -Wreturn-stack-address не является решением, так как это опасно в больших проектах.
Я заметил, что удаление ключевого слова const избавляет от предупреждения, но я не знаю почему и не хочу предполагать, что функтор получает только неконстантные диапазоны.
Вопросы: как исправить код, чтобы избавиться от предупреждения? Почему удаление const избавляет от предупреждения?
1 ответ
Это так.
//
// When storing transform iterators, operator[]()
// fails because it returns by reference. Therefore
// operator()() is provided for these cases.
//
Таким образом, вы должны быть в состоянии исправить это с
(v | boost::adaptors::transformed(tuple_get<0>{})) (0);
который возвращает abstract_value_type
(который является ссылкой только в том случае, если элементы являются абстрактными, массивом или функцией, в противном случае - значением типа).