标准库中的 boost::make_transform_iterator 相当于什么?

问题描述 投票:0回答:1

处理 const 向量时,以下方法不起作用:

const std::vector<std::string> v;
v.push_back("test"); // error: v cannot be modified

相反,您必须在构造向量的同一行上初始化向量。然而,即使有此限制,

boost::make_transform_iterator
也可以轻松地在将另一个向量的元素推入 v 之前对它们进行某些操作。在此示例中,
convert
是一个一元函数,它返回输入元素的转换版本:

auto beg = boost::make_transform_iterator(args.begin(), convert);
auto end = boost::make_transform_iterator(args.end(), convert);

const std::vector<const char*> vc { beg, end };

我查看了

<iterator>
中可用的功能,但没有看到等效的功能。它只是丢失了还是标准库没有它是有原因的?

c++ c++11 boost iterator
1个回答
11
投票

更新响应评论,在底部添加了c++20版本

对于 C++11 总是有 lambda 就地初始化技巧:

const auto vc = [&]{
    std::vector<const char*> tmp(v.size());
    std::transform(v.begin(), v.end(), tmp.begin(), convert);
    return tmp;
}();

const auto vc = [&]{
    std::vector<const char*> tmp;
    tmp.reserve(v.size());
    std::transform(v.begin(), v.end(), back_inserter(tmp), convert);
    return tmp;
}();

看到它在Coliru上直播

也就是说,我更喜欢 Boost Range 适配器:(也Live On Coliru

const auto vc = boost::copy_range<std::vector<const char*> >(v | transformed(convert));
#include <algorithm>
#include <vector>
#include <iterator>
#include <string>

#include <functional>
#include <iostream>

int main()
{
    const std::vector</* const */ std::string> v { "test", "goodbye" };

    auto convert = std::mem_fn(&std::string::c_str);

    const auto vc = [&]{
        std::vector<const char*> tmp;
        tmp.reserve(v.size());
        std::transform(v.begin(), v.end(), back_inserter(tmp), convert);
        return tmp;
    }();
    
    for (auto cc : vc)
        std::cout << cc << "\n";
}

更新:C++20 范围

2023 年将出现类似于 Boost 的标准化范围实用程序:Live On Coliru

#include <iostream>
#include <ranges>
#include <vector>

int main() {
    const std::vector<std::string> v{"test", "goodbye"};

    auto view = std::views::transform(v, &std::string::c_str);
    std::vector<char const*> vc(view.begin(), view.end());

    for (auto cc : vc)
        std::cout << cc << "\n";
}
© www.soinside.com 2019 - 2024. All rights reserved.