C++标准库学习之 STL Removing 操作

在研究 STL 的remove 操作时,有很多让我非常意外的问题,使用下面例子来说一下

c 复制代码
#include <iostream>
#include <list>
#include <algorithm>

int main(){


    // 创建一个向量,带有初始内容
    std::list<int> tsm_vector ={1,2,3,4,5,6,7,8,9};

    // 先删除  5 这个节点,删除成功后,尾结点会向前移动
    auto remove=    std::remove(tsm_vector.begin(),tsm_vector.end(),5);

    for (auto item:tsm_vector){
        std::cout<< item<<std::endl;
    }

    return 0;
}

结果:

vbnet 复制代码
D:\CWorkSpace\tsmTest\cmake-build-debug\tsmTest.exe
1
2
3
4
6
7
8
9
9

Process finished with exit code 0

在删除后发现所有节点都向前移动了,但是最后一个节点他没管,看了一下他的源码,发现他的源码就是不停的向后遍历,

那么该如何删除掉最后这个无用的9呢,

c 复制代码
#include <iostream>
#include <list>
#include <algorithm>

int main(){

    // 创建一个向量,带有初始内容
    std::list<int> tsm_vector ={1,2,3,4,5,6,7,8,9};

    // 先删除  5 这个节点,删除成功后,尾结点会向前移动
    auto remove=    std::remove(tsm_vector.begin(),tsm_vector.end(),5);

    // 在使用 erase 方法擦除无用节点
    tsm_vector.erase(remove,tsm_vector.end());
    
    for (auto item:tsm_vector){
        std::cout<< item<<std::endl;
    }

    return 0;
}

在使用完 remove 后,这个方法会返回他删除后最新的最后一个节点,也就是新的 end ,我们只需要使用 erase[remove,eng) 这个区间就可以了

那我们能直接使用 erase 直接参数吗, 答案肯定是可以的,

c 复制代码
#include <iostream>
#include <list>
#include <algorithm>

int main() {


    // 创建一个向量,带有初始内容
    std::list<int> tsm_vector = {1, 2, 3, 4, 5, 6, 7, 8, 9};

    // 先删除  5 这个节点,删除成功后,尾结点会向前移动
    auto remove = std::remove(tsm_vector.begin(), tsm_vector.end(), 5);

    // 在使用 erase 方法擦除无用节点
    tsm_vector.erase(remove, tsm_vector.end());

    for (auto item:tsm_vector) {
        std::cout << item << std::endl;
    }


    std::cout << "----------------使用 remove 后,在使用 erase-------------------" << std::endl;

    std::cout << "----------------直接使用 erase-------------------" << std::endl;

    std::list<int>::iterator start = tsm_vector.begin();
    
    while (start!=tsm_vector.end()){
        if (*start % 2 == 0) {
            // 使用擦出后的地址继续遍历,否则原来的start 会指向野指针,导致while 循环出错,
            start=tsm_vector.erase(start);
        } else {
            start++;
        }
    }

    for (auto item:tsm_vector) {
        std::cout << item << std::endl;
    }

    return 0;
}

为什么remove 无法删除数据而 erase 可以呢,在 STL 算法的设计初衷就是为了程序的扩展性等等情况考虑的, remove 等方法是与 容器脱离的,而erase 是容器自身的方法,容器自身肯定是知道容器内部的情况的,所以就形成了现在的情况

上面的例子我们使用的是 std::remove 方法,这个是 algorithm 算法库为我们提供的一个方法,想要更方便的移除该如何操作呢,

那就是调用容器的remove 方法

arduino 复制代码
std::list<int> tsm_vector = {1, 2, 3, 4, 5, 6, 7, 8, 9};

// 调用 tsm_vector 这个容器的 remove 方法即可快速的移除数据
tsm_vector.remove(5);
相关推荐
jiunian_cn1 小时前
【c++】【STL】list详解
数据结构·c++·windows·list·visual studio
虾球xz1 小时前
游戏引擎学习第250天:# 清理DEBUG GUID
c++·学习·游戏引擎
我命由我123451 小时前
STM32 开发 - stm32f10x.h 头文件(内存映射、寄存器结构体与宏、寄存器位定义、实现点灯案例)
c语言·开发语言·c++·stm32·单片机·嵌入式硬件·嵌入式
淋过很多场雨2 小时前
现代c++获取linux所有的网络接口名称
java·linux·c++
这个懒人4 小时前
C++后端服务器常见开发框架
c++·后端·框架
橘颂TA5 小时前
【C++】数据结构 九种排序算法的实现
数据结构·c++·排序算法
大魔王(已黑化)5 小时前
LeetCode —— 572. 另一棵树的子树
c语言·数据结构·c++·算法·leetcode·职场和发展
byte轻骑兵5 小时前
【C++类和数据抽象】消息处理示例(2)
开发语言·c++
千里镜宵烛5 小时前
C++ 红黑树
java·开发语言·c++
敲上瘾5 小时前
高并发内存池(五):性能测试与性能优化
c++·功能测试·缓存·性能优化·线程·高并发内存池