3.7 list容器

3.7.1 list基本概念

功能:将数据进行链式存储
链表(list)是一种物理存储单元上非连续的存储结构,数据元素的逻辑顺序是通过链表中的指针链接实现的
链表的组成:链表由一系列结点组成
结点的组成:一个是存储数据元素的数据域,另一个是存储下一个结点地址的指针域
STL中的链表是一个双向循环链表
由于链表的存储方式并不是连续的内存空间,因此链表list中的迭代器只支持前移和后移,属于双向迭代器
list的优点:

  • 采用动态存储分配,不会造成内存浪费和溢出
  • 链表执行插入和删除操作十分方便,修改指针即可,不需要移动大量元素

list的缺点:

  • 链表灵活,但是空间(指针域) 和 时间(遍历)额外耗费较大

List有一个重要的性质,插入操作和删除操作都不会造成原有list迭代器的失效,这在vector是不成立的。
总结:STL中List和vector是两个最常被使用的容器,各有优缺点

3.7.2 list构造函数

功能描述:

  • 创建list容器

函数原型:

  • list lst; //list采用采用模板类实现,对象的默认构造形式:
  • list(beg,end); //构造函数将[beg, end)区间中的元素拷贝给本身。
  • list(n,elem); //构造函数将n个elem拷贝给本身。
  • list(const list &lst); //拷贝构造函数。

示例:

  1. #include <list>
  2. void printList(const list<int>& L) {
  3. for (list<int>::const_iterator it = L.begin(); it != L.end(); it++) {
  4. cout << *it << " ";
  5. }
  6. cout << endl;
  7. }
  8. void test01()
  9. {
  10. list<int>L1;
  11. L1.push_back(10);
  12. L1.push_back(20);
  13. L1.push_back(30);
  14. L1.push_back(40);
  15. printList(L1);
  16. list<int>L2(L1.begin(),L1.end());
  17. printList(L2);
  18. list<int>L3(L2);
  19. printList(L3);
  20. list<int>L4(10, 1000);
  21. printList(L4);
  22. }
  23. int main() {
  24. test01();
  25. system("pause");
  26. return 0;
  27. }

总结:list构造方式同其他几个STL常用容器,熟练掌握即可

3.7.3 list 赋值和交换

功能描述:

  • 给list容器进行赋值,以及交换list容器

函数原型:

  • assign(beg, end); //将[beg, end)区间中的数据拷贝赋值给本身。
  • assign(n, elem); //将n个elem拷贝赋值给本身。
  • list& operator=(const list &lst); //重载等号操作符
  • swap(lst); //将lst与本身的元素互换。

示例:

  1. #include <list>
  2. void printList(const list<int>& L) {
  3. for (list<int>::const_iterator it = L.begin(); it != L.end(); it++) {
  4. cout << *it << " ";
  5. }
  6. cout << endl;
  7. }
  8. //赋值和交换
  9. void test01()
  10. {
  11. list<int>L1;
  12. L1.push_back(10);
  13. L1.push_back(20);
  14. L1.push_back(30);
  15. L1.push_back(40);
  16. printList(L1);
  17. //赋值
  18. list<int>L2;
  19. L2 = L1;
  20. printList(L2);
  21. list<int>L3;
  22. L3.assign(L2.begin(), L2.end());
  23. printList(L3);
  24. list<int>L4;
  25. L4.assign(10, 100);
  26. printList(L4);
  27. }
  28. //交换
  29. void test02()
  30. {
  31. list<int>L1;
  32. L1.push_back(10);
  33. L1.push_back(20);
  34. L1.push_back(30);
  35. L1.push_back(40);
  36. list<int>L2;
  37. L2.assign(10, 100);
  38. cout << "交换前: " << endl;
  39. printList(L1);
  40. printList(L2);
  41. cout << endl;
  42. L1.swap(L2);
  43. cout << "交换后: " << endl;
  44. printList(L1);
  45. printList(L2);
  46. }
  47. int main() {
  48. //test01();
  49. test02();
  50. system("pause");
  51. return 0;
  52. }

总结:list赋值和交换操作能够灵活运用即可

3.7.4 list 大小操作

功能描述:

  • 对list容器的大小进行操作

函数原型:

  • size(); //返回容器中元素的个数
  • empty(); //判断容器是否为空
  • resize(num); //重新指定容器的长度为num,若容器变长,则以默认值填充新位置。 //如果容器变短,则末尾超出容器长度的元素被删除。
  • resize(num, elem); //重新指定容器的长度为num,若容器变长,则以elem值填充新位置。 //如果容器变短,则末尾超出容器长度的元素被删除。

示例:

  1. #include <list>
  2. void printList(const list<int>& L) {
  3. for (list<int>::const_iterator it = L.begin(); it != L.end(); it++) {
  4. cout << *it << " ";
  5. }
  6. cout << endl;
  7. }
  8. //大小操作
  9. void test01()
  10. {
  11. list<int>L1;
  12. L1.push_back(10);
  13. L1.push_back(20);
  14. L1.push_back(30);
  15. L1.push_back(40);
  16. if (L1.empty())
  17. {
  18. cout << "L1为空" << endl;
  19. }
  20. else
  21. {
  22. cout << "L1不为空" << endl;
  23. cout << "L1的大小为: " << L1.size() << endl;
  24. }
  25. //重新指定大小
  26. L1.resize(10);
  27. printList(L1);
  28. L1.resize(2);
  29. printList(L1);
  30. }
  31. int main() {
  32. test01();
  33. system("pause");
  34. return 0;
  35. }

总结:

  • 判断是否为空 —- empty
  • 返回元素个数 —- size
  • 重新指定个数 —- resize

    3.7.5 list 插入和删除

    功能描述:

  • 对list容器进行数据的插入和删除

函数原型:

  • push_back(elem);//在容器尾部加入一个元素
  • pop_back();//删除容器中最后一个元素
  • push_front(elem);//在容器开头插入一个元素
  • pop_front();//从容器开头移除第一个元素
  • insert(pos,elem);//在pos位置插elem元素的拷贝,返回新数据的位置。
  • insert(pos,n,elem);//在pos位置插入n个elem数据,无返回值。
  • insert(pos,beg,end);//在pos位置插入[beg,end)区间的数据,无返回值。
  • clear();//移除容器的所有数据
  • erase(beg,end);//删除[beg,end)区间的数据,返回下一个数据的位置。
  • erase(pos);//删除pos位置的数据,返回下一个数据的位置。
  • remove(elem);//删除容器中所有与elem值匹配的元素。

示例:

  1. #include <list>
  2. void printList(const list<int>& L) {
  3. for (list<int>::const_iterator it = L.begin(); it != L.end(); it++) {
  4. cout << *it << " ";
  5. }
  6. cout << endl;
  7. }
  8. //插入和删除
  9. void test01()
  10. {
  11. list<int> L;
  12. //尾插
  13. L.push_back(10);
  14. L.push_back(20);
  15. L.push_back(30);
  16. //头插
  17. L.push_front(100);
  18. L.push_front(200);
  19. L.push_front(300);
  20. printList(L);
  21. //尾删
  22. L.pop_back();
  23. printList(L);
  24. //头删
  25. L.pop_front();
  26. printList(L);
  27. //插入
  28. list<int>::iterator it = L.begin();
  29. L.insert(++it, 1000);
  30. printList(L);
  31. //删除
  32. it = L.begin();
  33. L.erase(++it);
  34. printList(L);
  35. //移除
  36. L.push_back(10000);
  37. L.push_back(10000);
  38. L.push_back(10000);
  39. printList(L);
  40. L.remove(10000);
  41. printList(L);
  42. //清空
  43. L.clear();
  44. printList(L);
  45. }
  46. int main() {
  47. test01();
  48. system("pause");
  49. return 0;
  50. }

总结:

  • 尾插 —- push_back
  • 尾删 —- pop_back
  • 头插 —- push_front
  • 头删 —- pop_front
  • 插入 —- insert
  • 删除 —- erase
  • 移除 —- remove
  • 清空 —- clear

    3.7.6 list 数据存取

    功能描述:

  • 对list容器中数据进行存取

函数原型:

  • front(); //返回第一个元素。
  • back(); //返回最后一个元素。

示例:

  1. #include <list>
  2. //数据存取
  3. void test01()
  4. {
  5. list<int>L1;
  6. L1.push_back(10);
  7. L1.push_back(20);
  8. L1.push_back(30);
  9. L1.push_back(40);
  10. //cout << L1.at(0) << endl;//错误 不支持at访问数据
  11. //cout << L1[0] << endl; //错误 不支持[]方式访问数据
  12. cout << "第一个元素为: " << L1.front() << endl;
  13. cout << "最后一个元素为: " << L1.back() << endl;
  14. //list容器的迭代器是双向迭代器,不支持随机访问
  15. list<int>::iterator it = L1.begin();
  16. //it = it + 1;//错误,不可以跳跃访问,即使是+1
  17. }
  18. int main() {
  19. test01();
  20. system("pause");
  21. return 0;
  22. }

总结:

  • list容器中不可以通过[]或者at方式访问数据
  • 返回第一个元素 —- front
  • 返回最后一个元素 —- back

    3.7.7 list 反转和排序

    功能描述:

  • 将容器中的元素反转,以及将容器中的数据进行排序

函数原型:

  • reverse(); //反转链表
  • sort(); //链表排序

示例:

  1. void printList(const list<int>& L) {
  2. for (list<int>::const_iterator it = L.begin(); it != L.end(); it++) {
  3. cout << *it << " ";
  4. }
  5. cout << endl;
  6. }
  7. bool myCompare(int val1 , int val2)
  8. {
  9. return val1 > val2;
  10. }
  11. //反转和排序
  12. void test01()
  13. {
  14. list<int> L;
  15. L.push_back(90);
  16. L.push_back(30);
  17. L.push_back(20);
  18. L.push_back(70);
  19. printList(L);
  20. //反转容器的元素
  21. L.reverse();
  22. printList(L);
  23. //排序
  24. L.sort(); //默认的排序规则 从小到大
  25. printList(L);
  26. L.sort(myCompare); //指定规则,从大到小
  27. printList(L);
  28. }
  29. int main() {
  30. test01();
  31. system("pause");
  32. return 0;
  33. }

总结:

  • 反转 —- reverse
  • 排序 —- sort (成员函数)

    3.7.8 排序案例

    案例描述:将Person自定义数据类型进行排序,Person中属性有姓名、年龄、身高
    排序规则:按照年龄进行升序,如果年龄相同按照身高进行降序
    示例: ```cpp

    include

    include

    class Person { public: Person(string name, int age , int height) {
    1. m_Name = name;
    2. m_Age = age;
    3. m_Height = height;
    }

public: string m_Name; //姓名 int m_Age; //年龄 int m_Height; //身高 };

bool ComparePerson(Person& p1, Person& p2) {

  1. if (p1.m_Age == p2.m_Age) {
  2. return p1.m_Height > p2.m_Height;
  3. }
  4. else
  5. {
  6. return p1.m_Age < p2.m_Age;
  7. }

}

void test01() {

  1. list<Person> L;
  2. Person p1("刘备", 35 , 175);
  3. Person p2("曹操", 45 , 180);
  4. Person p3("孙权", 40 , 170);
  5. Person p4("赵云", 25 , 190);
  6. Person p5("张飞", 35 , 160);
  7. Person p6("关羽", 35 , 200);
  8. L.push_back(p1);
  9. L.push_back(p2);
  10. L.push_back(p3);
  11. L.push_back(p4);
  12. L.push_back(p5);
  13. L.push_back(p6);
  14. for (list<Person>::iterator it = L.begin(); it != L.end(); it++) {
  15. cout << "姓名: " << it->m_Name << " 年龄: " << it->m_Age
  16. << " 身高: " << it->m_Height << endl;
  17. }
  18. cout << "---------------------------------" << endl;
  19. L.sort(ComparePerson); //排序
  20. for (list<Person>::iterator it = L.begin(); it != L.end(); it++) {
  21. cout << "姓名: " << it->m_Name << " 年龄: " << it->m_Age
  22. << " 身高: " << it->m_Height << endl;
  23. }

}

int main() {

  1. test01();
  2. system("pause");
  3. return 0;

} ``` 总结:

  • 对于自定义数据类型,必须要指定排序规则,否则编译器不知道如何进行排序
  • 高级排序只是在排序规则上再进行一次逻辑规则制定,并不复杂