vector的erase()方法遍历删除元素迭代器失效问题、及删除最后一个元素迭代器失效问题)

1.删除指定范围的元素
vector删除元素之pop_back(),erase(),remove()
向量容器vector的成员函数pop_back()可以删除最后一个元素.

而函数erase()可以删除由一个iterator指出的元素,也可以删除一个指定范围的元素。

还可以采用通用算法remove()来删除vector容器中的元素.

不同的是:采用remove一般情况下不会改变容器的大小,而pop_back()与erase()等成员函数会改变容器的大小。

2.删除指定大小的元素
1. 方法一:
由于上面的删除方法都是只能删除指定的迭代器指定的位置元素,所以如果需要删除指定大小的元素则需要先找到其迭代器,不同于map(map有find方法),vector本身是没有find这一方法,其find是依靠algorithm来实现的。
给个例子:

#include <iostream>
#include <vector>
#include <algorithm>
using namespace std;

template<typename T>
inline void vectorDel(std::vector<T>& vec, T a) {
    auto it = std::find(vec.begin(), vec.end(), a);
    vec.erase(it);
}

int main()
{
    vector<int> vec;
    for(int i=0;i<10;i++){
        vec.push_back(i);
    }
    for(int i=0;i<vec.size();i++){
        cout << vec[i] <<' ';
    }
    cout<<endl;
    // 1.
    vector<int>::iterator it = find(vec.begin(), vec.end(), 2);
    // vec.erase(vec.begin()+5);//erase the 6th element
    vec.erase(it);
    // 2.
    vectorDel<int>(vec, 3);
    for(int i=0;i<vec.size();i++){
        cout << vec[i] << ' ';
    }
    cout<<endl;
    return 0;
}


运行结果:

0 1 2 3 4 5 6 7 8 9 
0 1 4 5 6 7 8 9

方法二:
摘抄自:对vector删除元素的效率比较
最近在看github上的开源代码,看到了vector删除元素的操作,如(1)所示
(1)、先用swap把要删除的元素和vector里最后一个元素交换位置,然后把最后一个元素pop_back

std::swap(*it, observers_.back());
observers_.pop_back();



我平时删除vector上的元素跟(2)是一样的
(2)、先用find查找元素,然后用erase删除元素

Iterator it = std::find(observers_.begin(), observers_.end(), x);
observers_.erase(it);



1的效率高,2删除元素后需要把后面的元素依次向前移动,但有时会要求不能改变vector中元素顺序,此时只能使用2
3.C++20 std::erase, std::erase_if (std::vector)
std::erase, std::erase_if (std::vector)中给了解释和例子。

注意点
erase()方法遍历删除某些元素迭代器失效问题、及删除最后一个元素迭代器失效问题)
给一个例子:

template<typename T = int>
bool remove_array(std::vector<T> &array, T v){
    typename std::vector<T>::iterator it = std::find(array.begin(), array.end(), v);
    if(it == array.end()){
        return false;
    }
    std::swap(*it, array.back());
    array.pop_back();
    return true;
}
void test_delete_in_vector() {
  // 例1
  std::vector<int> array;
  array.push_back(26);
  remove_array(array, 26);
  for (auto v : array) {
    std::cout << "array: " << v << std::endl;
  }
  // 例2
  std::vector<int> test_vec;
  for(int i = 0; i < 10; i++) {
    test_vec.emplace_back(i);
  }
  auto iter= test_vec.begin();
  while(iter!=test_vec.end()) {
      if((*iter) % 2 == 1) {
          iter = test_vec.erase(iter); // 如果删除的是最后一个元素,则返回的是迭代器为空,所以不能++
      } else {
          ++iter;
      }
  }
  for (auto v : test_vec) {
    std::cout << v << std::endl;
  }
}



4.删除重复元素

#include <iostream>
#include <vector>
#include <algorithm>

int main() {
    std::vector<int> numbers = {1, 2, 2, 3, 3, 3, 4, 5, 5, 6};

    // 使用 std::unique 去除相邻重复元素
    auto new_end = std::unique(numbers.begin(), numbers.end());

    // 使用容器的 erase 方法删除重复元素
    numbers.erase(new_end, numbers.end());

    // 打印结果
    std::cout << "Numbers after removing duplicates: ";
    for (const auto& num : numbers) {
        std::cout << num << " ";
    }
    std::cout << std::endl;

    return 0;
}


/*
output:
Numbers after removing duplicates: 1 2 3 4 5 6 
*/
 

1.vector迭代器失效问题
什么情况下会失效?以及失效的原因
vector是个连续内存存储的容器,如果vector容器的中间某个元素被删除或从中间插入一个元素, 有可能导致内存空间不够用而重新分配一块大的内存

造成失效的其中一原因:
        是因为内存的重新分配, 保留下来的迭代器不再指向容器中原来的元素

还有一种是删除元素,迭代器指向的空间自然就是一个无效的地址,无法再使用
使得vector迭代器失效的操作有:

(1)执行erase方法时,指向删除节点及其之后的全部迭代器均失效;
         

那么我们更新迭代器试试? ok了

但是,请看下图:一样的代码,只不过vector里的元素最后一个改为0了,报错了!

为什么报错呢?

原因:只有要删除的是最后一个元素,第一套代码才会暴露问题,因为删除最后一个元素,返回下一个迭代器是个空,对空指针进行++操作肯定崩溃

正确的写法如下:这样避免删除最后一个元素还对空迭代器操作

#include<vector>
class Solution {
public:
    int removeElement(vector<int>& nums, int val) {

        if(nums.size()>100)
        {
            cout<<"0 <= nums.length <= 100"<<endl;
            return nums.size();
        }
        if (val<0 || val>100)
        {
            cout<<"0 <= val <= 100"<<endl;
            return nums.size();
        }

		for (vector<int>::iterator iter = nums.begin(); iter != nums.end(); )
		{
			if (val == *iter)
			{
				swap(*iter,nums.back() );
				nums.pop_back();
			}
            else
            {
                iter++;
            }
			
		}

		return nums.size();

    }
};

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/272623.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

数字电子技术 一天速成

文章目录 一、数制与编码1. 数制转换2. BCD编码 二、逻辑代数1. 常见逻辑运算及逻辑门 三、化简逻辑表达式1. 卡诺图 求 表达式2. 表达式 画 卡诺图3. 卡诺图 化简 表达式4. 公式法 化简 表达式 ⭐⭐5. 表达式 求 反函数6. 卡诺图 求 反函数 四、组合逻辑电路的分析和设计1. 逻…

Mysql(5日志备份恢复)

一.日志管理 MySQL 的日志默认保存位置为 /usr/local/mysql/data 先看下mysql的日志文件有无&#xff1a; 修改配置文件添加&#xff1a;错误日志&#xff0c;用来记录当MySQL启动、停止或运行时发生的错误信息&#xff0c;默认已开启 修改配置文件添加&#xff1a;通用查…

.json文件转为.dll文件后还能读取吗?

(只是修改了后缀名而已&#xff0c;做一个伪装&#xff09; 测试&#xff1a; QFile file(QApplication::applicationDirPath() "/config.dll");qDebug()<<QApplication::applicationDirPath() "/config.dll";if (file.open(QIODevice::ReadOnly))…

设计模式--抽象工厂模式

实验4&#xff1a;抽象工厂模式 本次实验属于模仿型实验&#xff0c;通过本次实验学生将掌握以下内容&#xff1a; 1、理解抽象工厂模式的动机&#xff0c;掌握该模式的结构&#xff1b; 2、能够利用抽象工厂模式解决实际问题。 [实验任务]&#xff1a;人与肤色 使用抽象…

Netty—Reactor线程模型详解

文章目录 前言线程模型基本介绍线程模型分类Reactor线程模型介绍Netty线程模型&#xff1a; 传统阻塞IO的缺点Reactor线程模型单Reactor单线程模式单Reactor多线程模式主从Reactor多线程Reactor 模式小结 Netty 线程模型案例说明&#xff1a;Netty核心组件简介ChannelPipeline与…

服务运营 | 年终回顾:服务运营为您服务

文章作者&#xff1a;李舒湉&#xff0c;蔡君洋, Guo&#xff0c;陈盈鑫&#xff0c;王畅 编者按 在服务运营板块成立的第一年&#xff0c;给大家带来我们这一年中原创文章的年终回顾。迈向2024&#xff0c;服务运营继续为您服务 在服务运营板块成立的第一年&#xff0c;明确板…

预测块划分与亚像素精度:提升视频编码与图像处理的效率

在视频编码和图像处理中&#xff0c;预测块划分与亚像素精度是两项关键技术。本文将介绍预测块划分和亚像素精度的基本原理&#xff0c;探讨其在提高视频编码效率和图像处理精度方面的应用。 1. 预测块划分的基本原理 预测块划分是视频编码中的一项关键技术&#xff0c;它将图…

KCTF-Web-签到题

题目环境&#xff1a; 一道签到题 没有必要想那么麻烦 刚开始我以为是SQL注入 F12查源代码 在底部发现base64编码 进行base64解码ZmxhZ3t3ZTFjME1FX3RvXzB1Ul9jb050RVNUfQecho "ZmxhZ3t3ZTFjME1FX3RvXzB1Ul9jb050RVNUfQ" | base64 -d得到flag&#xff1a;flag{we1c0M…

spring状态机

1、概述 Spring State Machine 是一个用于处理状态机逻辑的框架&#xff0c;它提供了一种简洁的方法来定义状 态、转换以及在状态变更时触发的动作。 概念 状态 &#xff08; State &#xff09; &#xff1a;一个状态机至少要包含两个状态。例如自动门的例子&#xff0c;有 …

用芯片SIC8833可开发电子秤方案

SIC8833作为一款高性能的电子秤方案芯片&#xff0c;这款芯片是一个带24bitADC的8位RISC MCU&#xff0c;内置8k16位OTP程序存储器。具体24位双向I/O口的特性&#xff0c;广泛应用于电子衡器和精密测量及控制系统&#xff0c;能满足用户的不同需求和应用场景。 以下是电子秤方案…

盘古信息IMS-MOM制造运营管理系统,构建生产现场管理信息系统的最佳选择

在当今高度竞争的制造行业中&#xff0c;高效的生产管理是企业成功的关键。盘古信息IMS-MOM制造运营管理系统作为一款领先的管理系统其关键特性为制造企业构建生产现场管理信息系统提供了强大的优势。IMS-MOM不仅仅是一个软件系统&#xff0c;更是一种技术和管理手段的结合&…

【SQL经典题目】连续日期判断、同时在线人数、会话划分、间隔日期连续、日期交叉

【1.查询至少连续3天下单的用户】 思路1&#xff08;使用lead&#xff09;&#xff1a; distinct user_id,create_date去重&#xff0c;确保每个用户每天只有一条访问记录lead(create_date,2,‘9999-12-31’) over(partition by user_id order by create_date)根据用户分区&am…

力扣刷题记录(19)LeetCode:279、139

279. 完全平方数 这题和上篇文章的题类似&#xff0c;直接上代码 class Solution { public:int numSquares(int n) {vector<int> dp(n1,INT_MAX);dp[0]0;//j表示背包容量&#xff0c;dp[j]表示和为n的完全平方数的最少数量for(int i0;i*i<n;i){for(int ji*i;j<n;j…

【Java】springboot

文章目录 Spingboot1、起步依赖2、构建springboot工程jar包3、springboot配置文件4、多环境配置5、maven和boot多环境兼容问题6、配置文件分类7、springboot整合mybatis Spingboot springboot用来简化spring的初始搭建以及开发过程。 比方说&#xff0c;创建一个springmvc程序…

AI时代下,如何看待“算法利维坦”?

ChatGPT的浪潮从2022年袭来后&#xff0c;至今热度不减&#xff0c;呈现出蓬勃发展的趋势。AI家居、医疗、教育、金融、公益、农业、艺术…AI真的已经走进了生活的方方面面&#xff0c;我们仿佛已经进入了AI时代&#xff0c;势不可挡。人工智能水平如此之高&#xff0c;不禁感慨…

医疗器械行业为什么要搭建自己的知识付费平台

随着医疗技术的不断进步&#xff0c;医疗器械行业正迅速发展&#xff0c;成为全球范围内的热门产业。医疗器械行业需要不断更新技术、提升产品质量、加强用户培训和推广新产品。在这个过程中&#xff0c;搭建自己的知识付费平台变得越来越重要&#xff0c;本文将深入探讨为什么…

DAY1C++

1、思维导图 2.定义自己的命名空间myspace&#xff0c;并在myspace中定义一个字符串&#xff0c;实现求字符串大小的函数。 #include <iostream>using namespace std;namespace myspace{string s1("call your name");int len_s(string s){return s1.length();…

关于Word中隐藏文本的知识,看这篇就够了

Word允许你隐藏文本&#xff0c;这样你就可以阅读或打印文档&#xff0c;就好像文本不在那里一样。这似乎毫无意义&#xff0c;因为如果你不想让别人阅读&#xff0c;为什么不删除文本呢&#xff1f;但隐藏文本确实有一些有趣的用途。让我们来看看什么是隐藏文本&#xff08;什…

Python(六)—— 自定义模块

15. 自定义模块 15.1 模块的定义与分类 15.1.1 模块的定义 一个函数封装一个功能&#xff0c;当遇到众多函数时&#xff0c;将这些相同的功能封装到一个文件中&#xff0c;那么这个存储着很多常用的功能的py文件&#xff0c;就是模块。模块就是文件&#xff0c;存放一堆常用…

Oracle 19c OCP 082考场真题解析第16题

考试科目&#xff1a;1Z0-082 考试题量&#xff1a;90 通过分数&#xff1a;60% 考试时间&#xff1a;150min 本文为云贝教育郭一军guoyJoe原创&#xff0c;请尊重知识产权&#xff0c;转发请注明出处&#xff0c;不接受任何抄袭、演绎和未经注明出处的转载。【云贝教育】Orac…