【C++】list的使用(下)

简介: 【C++】list的使用(下)

4. 数据修改

由于链表结构的特殊性,我们可以很方便的头插,尾插或者在任意位置插入,所以在这里库里面对于数据插入提供了很多种方式


1.数据插入

1. push_back:尾插

c4ff6151ad01522634132638604b7462.png


2. push_front:头插

9d8d9c737c2426989eb344844bbe26c8.png


3.insert:在任意位置插入删除

6e4fec604a7dab772931d1cddf1aac55.png

void Test_insert()
{
  vector<int> v(5, 888);
  list<int> lt;
  //尾插
  lt.push_back(1);
  lt.push_back(2);
  lt.push_back(3);
  lt.push_back(4);
  lt.push_back(5);
  lt.push_back(6);
  auto it_out = lt.begin();
  while (it_out != lt.end())
  {
    cout << *it_out << " ";
    ++it_out;
  }
  cout << endl;
  //头插
  lt.push_front(10);
  it_out = lt.begin();
  while (it_out != lt.end())
  {
    cout << *it_out << " ";
    ++it_out;
  }
  cout << endl;
  //在任意位置插入
  auto it_push = ++lt.begin();//从第二个位置开始
  lt.insert(it_push, 30);//插入一个值
  it_out = lt.begin();
  while (it_out != lt.end())
  {
    cout << *it_out << " ";
    ++it_out;
  }
  cout << endl;
  ++it_push;
  lt.insert(it_push, 5, 50);//插入n个值
  it_out = lt.begin();
  while (it_out != lt.end())
  {
    cout << *it_out << " ";
    ++it_out;
  }
  cout << endl;
  ++it_push;
  lt.insert(it_push, v.begin(), v.end());//插入一个迭代器区间
  it_out = lt.begin();
  while (it_out != lt.end())
  {
    cout << *it_out << " ";
    ++it_out;
  }
  cout << endl;
}


240af744305ed8d89c59b7688d211f4d.png


2. 数据删除

与数据插入相对应的,数据删除也有三个

1. pop_back:尾删

f3de5b816274a3b01f30653223d2c50c.png


2.pop_front:头删

10f6d621146e980192e512cdafa52781.png


3.erase:任意位置删除

d315737d0195f6b4c0b6585ee6b70d24.png


void Test_erase()
{
  list<int> lt;
  for (int i = 0; i < 10; ++i)
  {
    lt.push_back(i);
  }
  auto it_out = lt.begin();
  while (it_out != lt.end())
  {
    cout << *it_out << " ";
    ++it_out;
  }
  cout << endl;
  //头删
  lt.pop_front();
  it_out = lt.begin();
  while (it_out != lt.end())
  {
    cout << *it_out << " ";
    ++it_out;
  }
  cout << endl;
  //尾删
  lt.pop_back();
  it_out = lt.begin();
  while (it_out != lt.end())
  {
    cout << *it_out << " ";
    ++it_out;
  }
  cout << endl;
  //任意位置删除
  auto pos = ++lt.begin();
  pos = lt.erase(pos);//删除某一位置
  it_out = lt.begin();
  while (it_out != lt.end())
  {
    cout << *it_out << " ";
    ++it_out;
  }
  cout << endl;
  auto start = ++pos;
  auto end = ++(++start);
  lt.erase(start, end);//删除一个迭代器区间
  it_out = lt.begin();
  while (it_out != lt.end())
  {
    cout << *it_out << " ";
    ++it_out;
  }
  cout << endl;
}

b8e8ff92d498662aa9600319d20ac06a.png


5.其他接口

除了上述的接口之外,还有一些我们之前在string和vector中没有见到的接口,下面我们来看看他们的用法

1. remove:删除list中指定值

4c1bcb45eb2c8a042d039132b22ab697.png

void Test_remove()
{
  list<int> lt;
  lt.push_back(1);
  lt.push_back(2);
  lt.push_back(3);
  lt.push_back(4);
  lt.push_back(5);
  lt.push_back(6);
  lt.push_back(7);
  lt.push_back(8);
  auto it_out = lt.begin();
  while (it_out != lt.end())
  {
    cout << *it_out << " ";
    ++it_out;
  }
  cout << endl;
  lt.remove(5);
  lt.remove(10);
  it_out = lt.begin();
  while (it_out != lt.end())
  {
    cout << *it_out << " ";
    ++it_out;
  }
  cout << endl;
}

39dadfc7716b1a609aa9d6832a7c7cca.png


可以看到,remove对于list中不存在的元素不会做任何操作

2. sort:排序list

4107be9707e30395b08615fd56c26cf1.png


看到这里肯定会有人有疑惑,sort算法库里面不是实现过吗?为什么又要重新在list里实现一下,我直接用算法库里面的不行吗?

✅答案是:不行,接下来看实验==》

8e81eeb344b8de8ea6a548962eabf847.png

可以看到,在编译的过程就已经报错了,这是因为库里面没有支持list迭代器类型的构造,为啥嘞?因为之前sort对容器内部的元素操作使用了+和-操作,但是list由于结构的限制,不支持迭代器的这个行为,所以对于list,要重新在库里面实现一个sort。


3. unique:删除list中的重复值

396dbe8769a04b11d007ba8a600615e0.png

这里注意一下,在使用unique之前要确保list是有序的,否则不能完成删除所有重复值的功能

void Test_Sort()
{
  list<int> lt;
  lt.push_back(1);
  lt.push_back(10);
  lt.push_back(9);
  lt.push_back(3);
  lt.push_back(6);
  lt.push_back(3);
  lt.push_back(7);
  lt.push_back(6);
  cout << "原list:";
  auto it_out = lt.begin();
  while (it_out != lt.end())
  {
    cout << *it_out << " ";
    ++it_out;
  }
  cout << endl;
  lt.unique();
  cout << "尝试在乱序的情况下使用unique:";
  it_out = lt.begin();
  while (it_out != lt.end())
  {
    cout << *it_out << " ";
    ++it_out;
  }
  cout << endl;
  lt.sort();
  cout << "排序list:";
  it_out = lt.begin();
  while (it_out != lt.end())
  {
    cout << *it_out << " ";
    ++it_out;
  }
  cout << endl;
  cout << "对有序的list使用unique:";
  lt.unique();
  it_out = lt.begin();
  while (it_out != lt.end())
  {
    cout << *it_out << " ";
    ++it_out;
  }
  cout << endl;
}

c4419961045a33dba05f8ad508f442bf.png

相关文章
|
4天前
|
编译器 C语言 C++
【c++丨STL】list模拟实现(附源码)
本文介绍了如何模拟实现C++中的`list`容器。`list`底层采用双向带头循环链表结构,相较于`vector`和`string`更为复杂。文章首先回顾了`list`的基本结构和常用接口,然后详细讲解了节点、迭代器及容器的实现过程。 最终,通过这些步骤,我们成功模拟实现了`list`容器的功能。文章最后提供了完整的代码实现,并简要总结了实现过程中的关键点。 如果你对双向链表或`list`的底层实现感兴趣,建议先掌握相关基础知识后再阅读本文,以便更好地理解内容。
14 1
|
17天前
|
算法 C语言 C++
【c++丨STL】list的使用
本文介绍了STL容器`list`的使用方法及其主要功能。`list`是一种双向链表结构,适用于频繁的插入和删除操作。文章详细讲解了`list`的构造函数、析构函数、赋值重载、迭代器、容量接口、元素访问接口、增删查改操作以及一些特有的操作接口如`splice`、`remove_if`、`unique`、`merge`、`sort`和`reverse`。通过示例代码,读者可以更好地理解如何使用这些接口。最后,作者总结了`list`的特点和适用场景,并预告了后续关于`list`模拟实现的文章。
33 7
|
25天前
|
存储 编译器 C++
C++ initializer_list&&类型推导
在 C++ 中,`initializer_list` 提供了一种方便的方式来初始化容器和传递参数,而右值引用则是实现高效资源管理和移动语义的关键特性。尽管在实际应用中 `initializer_list&&` 并不常见,但理解其类型推导和使用方式有助于深入掌握现代 C++ 的高级特性。
16 4
|
3月前
|
存储 搜索推荐 C++
【C++篇】深度剖析C++ STL:玩转 list 容器,解锁高效编程的秘密武器2
【C++篇】深度剖析C++ STL:玩转 list 容器,解锁高效编程的秘密武器
67 2
|
3月前
|
存储 算法 C++
【C++打怪之路Lv10】-- list
【C++打怪之路Lv10】-- list
24 1
|
3月前
|
存储 C++ 容器
【C++篇】深度剖析C++ STL:玩转 list 容器,解锁高效编程的秘密武器1
【C++篇】深度剖析C++ STL:玩转 list 容器,解锁高效编程的秘密武器
74 5
|
3月前
|
存储 编译器 C++
【C++篇】揭开 C++ STL list 容器的神秘面纱:从底层设计到高效应用的全景解析(附源码)
【C++篇】揭开 C++ STL list 容器的神秘面纱:从底层设计到高效应用的全景解析(附源码)
85 2
|
3月前
|
C++
【C++】C++ STL 探索:List使用与背后底层逻辑(三)
【C++】C++ STL 探索:List使用与背后底层逻辑
|
3月前
|
C++
【C++】C++ STL 探索:List使用与背后底层逻辑(二)
【C++】C++ STL 探索:List使用与背后底层逻辑
|
3月前
|
存储 编译器 C++
【C++】C++ STL 探索:List使用与背后底层逻辑(一)
【C++】C++ STL 探索:List使用与背后底层逻辑