C++如何使用std::copy_if来过滤容器元素?(代码示例)

std::copy_if可将满足条件的元素从源容器复制到目标容器而不修改源容器,需配合std::back_inserter确保目标容器动态扩容,谓词可用lambda等实现且不可修改元素。

使用 std::copy_if 可以把满足条件的元素从一个容器复制到另一个容器,不改变原容器,也不要求目标容器预先有足够空间(需确保目标迭代器可写入,比如用 std::back_inserter)。

基本用法:复制偶数到新 vector

需要头文件 。谓词(判断逻辑)可以是 lambda、函数指针或函数对象。

#include 
#include 
#include 
#include 

int main() {
    std::vector src = {1, 2, 3, 4, 5, 6};
    std::vector dst;

    // 复制所有偶数
    std::copy_if(src.begin(), src.end(),
                 std::back_inserter(dst),
                 [](int x) { return x % 2 == 0; });

    // 输出:2 4 6
    for (int x : dst) std::cout << x << " ";
}

复制到已有容器(需预留空间)

如果目标容器已存在且大小固定,必须保证容量足够,否则行为未定义。常用方式是先 resize 或用 reserve + back_inserter

std::vector src = {1, 2, 3, 4, 5};
std::vector dst;
dst.reserve(src.size()); // 预留空间避免多次重分配

std::copy_if(src.begin(), src.end(),
             std::back_inserter(dst),
             [](int x) { return x > 2; });
// dst 现在是 {3, 4, 5}

配合其他容器和自定义类型

std::copy_if 不限于 vector,也适用于 listdeque,甚至 C 数组;对结构体可按成员过滤。

立即学习“C++免费学习笔记(深入)”;

struct Person { std::string name; int age; };
std::vector people = {{"Alice", 25}, {"Bob", 17}, {"Charlie", 30}};

std::vector adults;
std::copy_if(people.begin(), people.end(),
             std::back_inserter(adults),
             [](const Person& p) { return p.age >= 18; });

注意事项

  • 源范围必须有效(begin ≤ end),目标迭代器必须支持写入
  • std::back_inserter 最安全,它自动调用 push_back
  • 若想就地移除元素,应使用 std::remove_if + erase,不是 copy_if
  • 谓词不能修改元素(应为 const 正确的 lambda 或函数)