gpt4 book ai didi

algorithm - 基于lambda的将向量拆分为多个较小向量的STL算法

转载 作者:塔克拉玛干 更新时间:2023-11-03 04:52:35 25 4
gpt4 key购买 nike

假设我有一个向量,其中包含一个结构,其中一个成员描述了它的目标向量。

struct Foo
{
int target;
static const int A = 0;
static const int B = 1;
static const int C = 2;
};

std::vector<Foo> elements;

std::vector<Foo> As;
std::vector<Foo> Bs;
std::vector<Foo> Cs;
std::vector<Foo> others;

现在我想根据 Target 的值将每个 Foo 移动到其他四个向量之一。

例如

auto elements = std::vector<Foo>{ {Foo::A}, {Foo::A}, {Foo::B} };

应该在 As 中产生两个元素,一个在 Bs 中,在 Csothers 中没有一个。 Elements 之后应该是空的。

我也可以自己做,但我想知道是否有一种 STL 算法可以用来完成它的工作。

最佳答案

标准算法通常不会对多个输出目标进行操作,因此当您想通过输出迭代器抽象出目标容器时,很难在这里找到合适的解决方案。可能最接近的是 std::copy_if。这可能看起来像

// Help predicate creation:
auto pred = [](int target){ return [target](const Foo& f){ return f.target == target; }; };

std::copy_if(elements.begin(), elements.end(), std::back_inserter(As), pred(Foo::A));
std::copy_if(elements.begin(), elements.end(), std::back_inserter(Bs), pred(Foo::B));
std::copy_if(elements.begin(), elements.end(), std::back_inserter(Cs), pred(Foo::C));
std::copy_if(elements.begin(), elements.end(), std::back_inserter(others),
[](const Foo& f){ return false; /* TODO */ });

elements.clear();

如果复制比移动构造更昂贵,您应该传递 std::make_move_iterator(elements.begin()) 并且 elements.end() 也是如此到算法。这里的问题是这不能扩展。 std::copy_if 线性遍历输入范围,上面要这样做四次。可以获得一次遍历,例如像下面这样。

auto doTheWork = [&As, &Bs, &Cs, &others](const Foo& foo) {
if (foo.target == Foo::A)
As.push_back(foo);
else if (foo.target == Foo::B)
Bs.push_back(foo);
else if (foo.target == Foo::C)
Cs.push_back(foo);
else
others.push_back(foo);
};

std::for_each(elements.begin(), elements.end(), doTheWork);

在这种情况下,我们至少采用了标准算法,但将逻辑转换为相当丑陋的 lambda。请注意,上面的 lambda 将始终复制其参数,需要进行一些调整才能正确使用 std::move_iterator

有时,一个很好的基于 for 循环的旧范围是最可读的解决方案。

关于algorithm - 基于lambda的将向量拆分为多个较小向量的STL算法,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/55039685/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com