当前位置: 首页 > news >正文

跟我学C++中级篇——STL的中的删除

一、介绍

在STL中一般删除的方式有两类,一种是使用全局的std::remove(remove_if类似),一种是使用容器自带的erase,前者其实并没有真正的删除数据,而后者则是在移动时,会有一些细节的处理,否则要么程序崩溃,要么达不到删除的目的。下面就这两个函数进行一下分析说明。
在一些容器中也提供了remove(如std::list)方法,它和全局的std::remove还是用法区别不小的,这个大家需要自己看一下。
注:C++20提供了std::erase这种更方便的用法

二、std::remove和 erase

1、std::remove
它的定义很简单:

template< class ForwardIt, class T >
ForwardIt remove( ForwardIt first, ForwardIt last, const T& value );(C++20)
template< class ForwardIt, class T >
constexpr ForwardIt remove( ForwardIt first, ForwardIt last, const T& value );(C++20)
template< class ExecutionPolicy, class ForwardIt, class T >
ForwardIt remove( ExecutionPolicy&& policy,ForwardIt first, ForwardIt last, const T& value );

其功能是实现删除在迭代器指定的范围内的所有满足条件的元素并返回尾后迭代器。需要说明的,这里的删除并未真正删除,只是移动,可以在下面的例程中看到相关的示例代码。
2、erase
看一下vector中的定义:

iterator erase( iterator pos );(until C++11)
iterator erase( const_iterator pos );(since C++11)(until C++20)
constexpr iterator erase( const_iterator pos );(since C++20)(2) 	
iterator erase( iterator first, iterator last );(until C++11)
iterator erase( const_iterator first, const_iterator last );(since C++11)(until C++20)
constexpr iterator erase( const_iterator first, const_iterator last );(since C++20)

它的定义很明显,有两种用法,一种是删除指定位置的元素,一种是删除指定范围的元素。这次之所以总结一下,就是因为发现错误的删除方式不再崩溃。所以得到的结论是,有时崩溃,有时不崩溃,看环境。

3、二者混合
其实就是移动并删除被删除的元素,有点拗口。就是将remove移动后的元素,不再使用的空间内的元素真正删除并和迭代器等自然挂钩。看下面的例子就会明白。

三、例程

例程是以std::vector做为例程的,其它的容器可能细节上会有一些差别,但总体上的应用是一致的。感兴趣可以把其它的几个容器都试一下。

#include <algorithm>
#include <cctype>
#include <iostream>
#include <string>
#include <vector>void testErase2() {std::vector<int> vec{1, 5, 2, 2, 3, 4, 2, 5, 6, 8, 9, 12};vec.erase(vec.begin() + 9, vec.end());std::cout << "cur vec:";for (auto &d : vec) {std::cout << d << ",";}std::cout << std::endl;
}
void testRemoveErase() {std::vector<int> vec{1, 5, 2, 2, 3, 4, 2, 5, 6, 8, 9, 12};auto it = std::remove(vec.begin(), vec.end(), 2);for (auto beg = it; beg != vec.end(); beg++) {std::cout << *beg << "---" << std::endl;}for (auto beg = vec.begin(); beg != vec.end(); beg++) {std::cout << *beg << "---" << std::endl;}vec.erase(it, vec.end());std::cout << "cur vec:";for (auto &d : vec) {std::cout << d << ",";}std::cout << std::endl;
}
void testDelVec() {std::vector<int> vec{1, 5, 5, 4, 5, 6};std::cout << "vec size:" << vec.size() << std::endl;for (auto it = vec.begin(); it != vec.end();) {if (*it == 5) {it = vec.erase(it);//vec.erase(it);//在Ubuntu22,Qt环境,不崩溃std::cout << "delete it:" << *it << std::endl;} else {it++;}}std::cout << "delete it end" << std::endl;std::cout << "delete it end vec value:" << std::endl;for (auto &d : vec) {std::cout << d << ",";}std::cout << std::endl;
}
void testVec() {std::vector<int> vec{1, 5, 5, 4, 5, 6};auto iter = std::remove(vec.begin(), vec.end(), 5);std::cout << "size is :" << vec.size() << std::endl;std::cout << "capacity is :" << vec.capacity() << std::endl;for (auto first = vec.begin(); first < iter; ++first) {std::cout << *first << " ";}return;
}
void testStr() {std::string test_str = "My test this   delete ORC";auto it = std::remove(test_str.begin(), test_str.end(), ' ');std::cout << test_str << std::endl;auto x = test_str.erase(it, test_str.end());std::cout << "test_str:" << test_str << std::endl;
}
int main() {testErase2();testRemoveErase();testDelVec();testVec();testStr();return 0;
}

另外还有一些不太标准的删除方式,如使用resize()函数等,一般不推荐,不过在某些场合下,可能会用着更方便。实际场景决定实际开发的代码吧,不要刻意追求某种方法。

四、总结

之所以把STL中的删除分析一下,是因为早期的印象是只要使用循环遍历的方式(不处理迭代器)暴力删除容器内容一定会崩溃,可现在发现在g++中其实是不崩溃的,把这两个和删除相关的函数就写了个程序跑了一下,给大家一个借鉴。没有什么难度,重点是细节要弄清楚。

http://www.lryc.cn/news/326475.html

相关文章:

  • js如何遍历查询一个颗树
  • 【面试必备】针对一个案例,怎么测试
  • vue3 hooks之事件广播(支持跨标签页)
  • go中validate包使用教程
  • canvas画带透明度的直线和涂鸦
  • linux命令 curl忽略https证书
  • 游戏引擎中网络游戏的基础
  • ES6(ECMAScript 6)中常用的知识点总结(包含示例代码)
  • 老师人手必备的教学神器有哪些?这5款教学软件一定要知道!
  • 华为机试题-核酸检测人数
  • SQLAlchemy模型映射提示declarative_base() takes 0 positional arguments but 1 was given
  • linux系统Kubernetes工具ingress暴露服务
  • centos2anolis
  • Cesium安装部署运行
  • 【Android 内存优化】KOOM线程泄漏监控的实现源码分析
  • 【爬虫基础】第1讲 网络爬虫基本知识
  • scrapy爬虫框架
  • 【深度学习】基础知识
  • Electron应用自动更新实现及打包部署全攻略
  • 【爬虫基础】第6讲 opener的使用
  • Milvus 向量数据库:如何基于docker-compose在本地快速搭建测试环境
  • python --dejavu音频指纹识别
  • 完全二叉树的层序遍历[天梯赛]
  • C语言看完我这篇编译与链接就够啦!!!
  • 【React】react 使用 lazy 懒加载模式的组件写法,外面需要套一层 Loading 的提示加载组件
  • IDEA的Scala环境搭建
  • LeetCode第四天(448. 找到所有数组中消失的数字)
  • 【vivado】在原有工程上新建工程
  • (原型与原型链)前端八股文修炼Day5
  • 逐步学习Go-并发通道chan(channel)