c++ - 如何在 C++ 中实现类似 for_each 的函数?

标签 c++ vector stl-algorithm

我需要实现一个 for_each 函数,如下所示。我知道 std::for_each 可以将 fn 应用于每个元素,但我们不能删除 std::for_each 中的元素。我需要扩展这个模板函数,以便fn 中,调用者可以同时访问元素和一次删除一个元素。有没有正确的方法来做到这一点?

#include <iostream>
#include <vector>
#include <algorithm>
using namespace std;
class A
{
public:
    explicit A(){
        mVec.clear();
    }
    ~A(){}
    template<class T> void for_each(T fn)
    {
        for(size_t i = 0; i < mVec.size(); ++i)
        {
            //fn can erase element or just visit element
            fn(mVec[i]);
        }
    }
    vector<int> mVec;
};
int main()
{
    A test;
    for(int i = 0; i < 8; ++i)
    {
        test.mVec.push_back(i);
    }
    test.for_each([&test](int i){
        if (i % 2 == 0)
        {
            cout << i << " deleted" << endl;
            test.mVec.erase(find(test.mVec.begin(), test.mVec.end(), i));
        } 
        else
        {
            cout << i << " parse" << endl;
        }
    });

    system("pause");
    return 0;
}

编辑:在for_each 模板函数中,我们不知道调用者是否会删除元素。删除元素是在 fn

中完成的

最佳答案

您能否从该函数返回一个 bool 值,其中 true 表示“删除元素”?然后你的 for_each 函数变成类似的东西。

    size_t i = 0;
    for(size_t j = 0; j < mVec.size(); ++j) {
        if (!fn(mVec[j])) {
            // The element must be kept
            if (i != j)
                mVec[i] = std::move(mVec[j]);
            i++;
        }
    }
    mVec.resize(i);

优点还在于,无论删除多少元素,这始终是 O(n)。

编辑:上面的循环实际上只是 std::remove_if(),所以 @ChenOT 的建议是最好的。或者

    n = std::remove_if(mVec.begin(), mVec.end(), fn) - mVec.begin();
    mVec.resize(n);

关于c++ - 如何在 C++ 中实现类似 for_each 的函数?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34018221/

相关文章:

c++ - 使用 copy 和 back_inserter 将 vector 附加到自身时的错误结果

c++ - '_IsFirstIteration':std::lower_bound 中未引用的形参

c++ - _block_type_is_valid(phead- nblockuse) 从 shared_ptr 的 vector 中删除数据时

c++ - 使用模板将一对 vector 中的所有元素合并为一个 vector

c++ - Boost 分配器的 vector 大小不正确

c++ - 如何计算给定 vector 索引的补码?

c++ - 在类成员函数中声明和定义函数对象

c++ - 如何简化我的 C++ 代码以反转字符?

c++ - 按位运算截断数字的最后两位

C++ 仅当模板参数为真时才创建变量