• list(链表)——STL


    list

    将数据进行链式存储
    链表(list)是一种物理存储单元上非连续的存储结构,数据元素的逻辑顺序是通过链表中的指针链接实现的
    链表的组成:链表由一系列结点组成
    结点的组成:一个是存储数据元素的数据域,另一个是存储下一个结点地址的指针域

    由于链表的存储方式并不是连续的内存空间,因此链表list中的迭代器只支持前移和后移,属于双向迭代器
    list的优点:

    • 采用动态存储分配,不会造成内存浪费和溢出
    • 链表执行插入和删除操作十分方便,修改指针即可,不需要移动大量元素
      list的缺点:
    • 链表灵活,但是空间(指针域) 和 时间(遍历)额外耗费较大

    List有一个重要的性质,插入操作和删除操作都不会造成原有list迭代器的失效,这在vector是不成立的。

    list构造函数

    创建list容器

    • list lst; //list采用采用模板类实现,对象的默认构造形式:
    • list(beg,end); //构造函数将[beg, end)区间中的元素拷贝给本身。
    • list(n,elem); //构造函数将n个elem拷贝给本身。
    • list(const list &lst); //拷贝构造函数。
    list<int>L1;
    L1.push_back(10);
    L1.push_back(20);
    L1.push_back(30);
    L1.push_back(40);
    printList(L1);
    list<int>L2(L1.begin(),L1.end());
    printList(L2);
    list<int>L3(L2);
    printList(L3);
    list<int>L4(10, 1000);
    printList(L4);
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12

    3 list 赋值和交换

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

    • assign(beg, end); //将[beg, end)区间中的数据拷贝赋值给本身。
    • assign(n, elem); //将n个elem拷贝赋值给本身。
    • list& operator=(const list &lst); //重载等号操作符
    • swap(lst); //将lst与本身的元素互换
    赋值操作
    
    //赋值
    list<int>L2;
    L2 = L1;
    printList(L2);
    list<int>L3;
    L3.assign(L2.begin(), L2.end());
    printList(L3);
    list<int>L4;
    L4.assign(10, 100);
    printList(L4);
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    交换
    
    list<int>L1;
    L1.push_back(10);
    L1.push_back(20);
    L1.push_back(30);
    L1.push_back(40);
    list<int>L2;
    L2.assign(10, 100);
    cout << "交换前: " << endl;
    printList(L1);
    printList(L2);
    cout << endl;
    L1.swap(L2);
    cout << "交换后: " << endl;
    printList(L1);
    printList(L2);
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18

    list 大小操作

    对list容器的大小进行操作

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

    list 插入和删除

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

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

    • 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值匹配的元素。

    list 数据存取

    对list容器中数据进行存取

    • list容器中不可以通过[]或者at方式访问数据

    • 返回第一个元素 — front

    • 返回最后一个元素 — back

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

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

    list<int>L1;
    L1.push_back(10);
    L1.push_back(20);
    L1.push_back(30);
    
    //cout << L1.at(0) << endl;	//错误 不支持at访问数据
    //cout << L1[0] << endl; 	//错误 不支持[]方式访问数据
    cout << "第一个元素为: " << L1.front() << endl;
    cout << "最后一个元素为: " << L1.back() << endl;
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9

    list容器的迭代器是双向迭代器,不支持随机访问

    //list容器的迭代器是双向迭代器,不支持随机访问
    list<int>::iterator it = L1.begin();
    //it = it + 1;//错误,不可以跳跃访问,即使是+1
    it++;  // 可以自增
    
    • 1
    • 2
    • 3
    • 4

    list反转和排序

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

    • reverse(); //反转链表
    • sort(); //链表排序(默认从小到大)
    //反转和排序
    void test01()
    {
    	list<int> L;
    	L.push_back(90);
    	L.push_back(10);
    	L.push_back(60);
    	printList(L);
    	//反转容器的元素
    	L.reverse();
    	printList(L);
    	//排序
    	L.sort(); //默认的排序规则 从小到大
    	printList(L);
    	
    	L.sort(myCompare); //指定规则,从大到小
    	printList(L);
    }
    
    bool myCompare(int val1 , int val2)
    {
    	return val1 > val2;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 对于自定义数据类型,必须要指定排序规则,否则编译器不知道如何进行排序
    • 高级排序只是在排序规则上再进行一次逻辑规则制定,并不复杂
    #include 
    #include 
    class Person {
    	public:
    	Person(string name, int age , int height) {
    	m_Name = name;
    	m_Age = age;
    	m_Height = height;
    }
    public:
    	string m_Name; //姓名
    	int m_Age; //年龄
    	int m_Height; //身高
    };
    bool ComparePerson(Person& p1, Person& p2) {
    	if (p1.m_Age == p2.m_Age) {
    		return p1.m_Height > p2.m_Height;
    	}
    	else
    		{
    			return p1.m_Age < p2.m_Age;
    		}
    	}
    void test01() {
    	list<Person> L;
    	Person p1("刘备", 35 , 175);
    	Person p2("曹操", 45 , 180);
    	Person p3("孙权", 40 , 170);
    	Person p4("赵云", 25 , 190);
    	Person p5("张飞", 35 , 160);
    	Person p6("关羽", 35 , 200);
    	L.push_back(p1);
    	L.push_back(p2);
    	L.push_back(p3);
    	L.push_back(p4);
    	L.push_back(p5);
    	L.push_back(p6);
    	
    for (list<Person>::iterator it = L.begin(); it != L.end(); it++) 
    {
    		cout << "姓名: " << it->m_Name << " 年龄: " << it->m_Age << " 身高: " << it->m_Height << endl;
    }
    	cout << "---------------------------------" << endl;
    	
    	L.sort(ComparePerson); //排序
    	
    	for (list<Person>::iterator it = L.begin(); it != L.end(); it++) 
    	{
    		cout << "姓名: " << it->m_Name << " 年龄: " << it->m_Age << " 身高: " << it->m_Height << endl;
    	}
    }
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
  • 相关阅读:
    Chrome速度无人能敌?Safari也甘拜下风
    工控机通过485modbus转profinet网关与温度智能表通讯案例
    Nios II 实现流水灯实验
    Anaconda成功安装之后没有在菜单列和桌面显示图标
    测试用例设计方法六脉神剑——第二剑:招式组合,因果判定出世
    uniapp-chatgpt跨端仿ChatGPT实例|uniapp+vue3+pinia多端聊天模板
    使用RAII+接口模式对模型加载和推理进行封装
    LabVIEW光电在线测振系统
    Linux系统管理
    NFT 推荐|Funguys 王国作品集第一弹
  • 原文地址:https://blog.csdn.net/mankeywang/article/details/126112520