std :: transform with std :: reference_wrapper

时间:2018-02-21 16:47:53

标签: c++ c++11

如何将std::transform与包含std::reference_wrapper类型对象的容器结合使用?从const std::vector<std::reference_wrapper<const float>>阅读并写入std::vector<float>时,一切正常,但如果我尝试写入std::vector<std::reference_wrapper<float>>,则不会编译给出错误

  

/ usr / include / c ++ / 5 / bits / stl_algo.h:4214:错误:使用已删除的函数&gt;&#39; std :: reference_wrapper&lt; _Tp&gt; :: reference_wrapper(_Tp&amp;&amp;)[with _Tp = float]&#39;     __ result = __binary_op( __ first1,* __ first2);

我认为它试图替换引用而不是更改引用值的值。有没有办法使用transform来实现这个目标,还是我更喜欢编写自己的函数?

编辑:添加了一个参考示例:

#include <iostream>
#include <vector>
#include <functional>
#include <algorithm>

int main()
{
std::vector<float> v(3, 1.0f);
std::vector<std::reference_wrapper<float>> refVector;

refVector.reserve(v.size());
for(auto& elem : v)
{
refVector.push_back((std::reference_wrapper<float>)elem);
}

std::vector<float> v2(3, 2.0f);
std::vector<float> v3(3, 3.0f);
std::vector<float> v4(3);

for(auto& elem : v)
std::cout << elem << std::endl; // all 1s

std::transform(refVector.begin(), refVector.end(), v3.begin(), v4.begin(), std::plus<float>());

for(auto& elem : v4)
std::cout << elem << std::endl; // all 4s

std::transform(v2.begin(), v2.end(), v3.begin(), refVector.begin(), std::minus<float>()); // doesn't compile

for(auto& elem : v)
std::cout << elem << std::endl; // want all -1s

return 0;
}

1 个答案:

答案 0 :(得分:1)

如果您要覆盖float中引用的现有std::vector<std::reference_wrapper<float>>,则需要将std::reference_wrapper<float>解包为float &的内容。 boost::indirect_iterator是这样的事情,虽然如果你不能添加一个提升依赖,很容易编写你自己的迭代器适配器。

示例用法:

std::vector<float> src1 = /* values */, src2 = /* other values */;
std::vector<std::reference_wrapper<float>> dest = /* references to somewhere */;

boost::indirect_iterator<std::reference_wrapper<float>, float> indirected(dest.begin());
std::transform(src1.begin(), src1.end(), src2.begin(), indirected, std::plus<float>());

草图实现

struct indirect_iterator
{
    /* a bunch of typedefs go here */
    indirect_iterator& operator++() { ++inner; }
    float & operator*() { return inner->get(); }
    float * operator->() { return &inner->get(); }  
private:
    std::vector<std::reference_wrapper<float>>::iterator inner;
}