boost :: range library

时间:2015-05-26 13:29:16

标签: c++ c++11 boost c++14 boost-range

我正在使用boost :: range库编写过滤器和地图算法:

template <class Range> struct Converter
{
    Converter(const Range& p_range) : m_range(p_range) {}

    template<class OutContainer> operator OutContainer() const
    {
        return {m_range.begin(), m_range.end()};
    }

private:
    Range m_range;
};

template<class Range> Converter<Range> convert(const Range& p_range) { return {p_range}; }

template<class Range, class Fun> auto map(Range&& p_range, Fun&& p_fun)
{
    return convert(p_range | boost::adaptors::transformed(p_fun));
}

template<class Range, class Pred> auto filter(Range&& p_range, Pred&& p_pred)
{
    return convert(p_range | boost::adaptors::filtered(p_pred));
}

现在我可以像这样使用它们:

std::vector<int> l_in  = {1, 2, 3, 4, 5};
std::vector<int> l_tmp_out = filter(l_in, [](int p){ return p < 4; });
std::vector<int> l_out = map(l_tmp_out, [](int p){ return p + 5; });

我也想以这种方式编写代码:

map(filter(l_in, [](int p){ return p < 4; }), [](int p){ return p + 5; });

不幸的是我的Converter类没有使用boost :: range算法,因此这个例子不能编译。我正在寻找一种改变它的正确方法。

更新

我关注了@sehe链接,事实证明我所要做的就是将这四行添加到Converter类中:

using iterator = typename Range::iterator;
using const_iterator = typename Range::const_iterator;
auto begin() const { return m_range.begin(); }
auto end() const { return m_range.end(); }

1 个答案:

答案 0 :(得分:2)

这是我对事物的看法:

<强> Live On Coliru

#include <boost/range.hpp>
#include <boost/range/adaptors.hpp>
#include <boost/range/algorithm.hpp>
#include <iostream>
#include <vector>

namespace MyRange {
    template <typename R> struct Proxy {
        Proxy(R&& r)      : _r(std::move(r)) {}
        Proxy(R const& r) : _r(r) {}

        template <typename OutContainer> operator OutContainer() const {
            return boost::copy_range<OutContainer>(_r);
        }

        using iterator       = typename boost::range_mutable_iterator<R>::type;
        using const_iterator = typename boost::range_const_iterator<R>::type;

        auto begin() const { return range_begin(_r); }
        auto end()   const { return range_end(_r);   }
        auto begin()       { return range_begin(_r); }
        auto end()         { return range_end(_r);   }

      private:
        R _r;
    };

    template <typename R> auto make_proxy(R&& r) { return Proxy<R>(std::forward<R>(r)); }

    template <typename Range, typename Fun> auto map(Range&& p_range, Fun&& p_fun) {
        return make_proxy(std::forward<Range>(p_range) | boost::adaptors::transformed(std::forward<Fun>(p_fun)));
    }

    template <typename Range, typename Pred> auto filter(Range&& p_range, Pred&& p_pred) {
        return make_proxy(std::forward<Range>(p_range) | boost::adaptors::filtered(std::forward<Pred>(p_pred)));
    }
}

int main() {
    using namespace MyRange;
    {
        std::vector<int> l_in  = {1, 2, 3, 4, 5};
        std::vector<int> l_tmp_out = filter(l_in, [](int p){ return p < 4; });
        std::vector<int> l_out = map(l_tmp_out, [](int p){ return p + 5; });

        boost::copy(l_out, std::ostream_iterator<int>(std::cout << "\nfirst:\t", "; "));
    }

    {
        boost::copy(
                map(
                    filter(
                        std::vector<int> { 1,2,3,4,5 },
                        [](int p){ return p < 4; }),
                    [](int p){ return p + 5; }),
                std::ostream_iterator<int>(std::cout << "\nsecond:\t", "; "));
    }
}

打印

first:  6; 7; 8; 
second: 6; 7; 8; 

注意

  • 更准确地使用std::forward<>
  • 它使用const / non-const迭代器
  • 它使用Boost Range特征(range_mutable_iterator<>等)而不是硬编码假定嵌套的typedef。这样就可以处理其他范围(例如std::array<>甚至int (&)[])。

  • 用户定义的转化运算符出于类似原因使用boost::copy_range<>