【C++】手撕 list类(包含迭代器)

news2025/4/4 4:19:58

目录

1,list的介绍及使用

2,list_node

3,list_node()

3,list

4,list()

5,push_back(const T& x)

6,print()

7,_list_iterator

8,operator*()

9,begin()

10,end()

11,operator->()

12,operator++()

13,operator++(int)

14,operator--()

15,operator--(int)

16,operator==(const sefl& s)

17,operator!=(const sefl& s)

18,_list_const_iterator

19,list(iterator first, iterator last)

20,begin()const

21,end()const

22,list(const list& lt)

23,operator=(list lt)

24,insert(iterator pos, const T& x)

25,erase(iterator pos)

26,clear()

27,~list()

28,push_front(const T& x)

29,pop_front()

30,pop_back()

31,源代码

32,总结


1,list的介绍及使用

1,list 是可以在常数范围内在任意位置进行插入和删除的序列式容器,并且该容器可以前后双向迭代。

2,list 的底层是双向链表结构,双向链表中每个元素存储在互不相关的独立节点中,在节点中通过指针指向其前一个元素和后一个元素。

3,list 与 forward_list 非常相似:最主要的不同在于 forward_list 是单链表,只能朝前迭代,已让其更简单高效。

4,与其他的序列式容器相比(array,vector,deque),list通常在任意位置进行插入、移除元素的执行效率更好。

5,与其他序列式容器相比,list 和 forward_list 最大的缺陷是不支持任意位置的随机访问,比如:要访问 list 的第6个元素,必须从已知的位置(比如头部或者尾部)迭代到该位置,在这段位置上迭代需要线性的时间开销;list 还需要一些额外的空间,以保存每个节点的相关联信息(对于存储类型较小元素的大list来说这可能是一个重要的因素)

2,list_node

结点的结构体框架

template<class T>
	struct list_node
	{
		list_node<T>* _next;
		list_node<T>* _prev;
		T data;
	};

因为是双向循环链表,需要一个上指针 _prev,下指针 _next,还有数据 data;

3,list_node()

对结点进行初始化

		list_node(const T& x = T())
			:_next(nullptr)
			, _prev(nullptr)
			, data(x)
		{}

然后还要将其初始化,指针为空,数据为内置类型初始化的值;

3,list

链表结构框架

	template <class T>
	class list
	{
		typedef list_node<T> node;
	public:

	private:
		node* _head;
	};

链表是带头结点的,所以我们需要一个哨兵位头结点;

4,list()

对链表初始化

		void empty_init()
		{
			_head = new node;
			_head->_next = _head;
			_head->_prev = _head;
		}

		list()
		{
			empty_init();
		}

因为我们是双向循环链表,所以我们的下一个结点和上一个结点都是指向自己的,形成一个环;

5,push_back(const T& x)

尾插

		void push_back(const T& x)
		{
			node* tail = _head->_prev;
			node* newnode = new node(x);

			tail->_next = newnode;
			newnode->_prev = tail;
			newnode->_next = _head;
			_head->_prev = newnode;
		}

我们先找到尾结点(tail),申请一个新结点,然后就插入其中;

6,print()

打印数据

		void print()
		{
			node* cur = _head->_next;
			while (cur != _head)
			{
				cout << cur->data << " ";
				cur = cur->_next;
			}
		}

哨兵位头结点本身是没有数据的,所以要从下一个结点开始

	void test1()
	{
		wxd::list<int> lt1;
		lt1.push_back(1);
		lt1.push_back(2);
		lt1.push_back(3);
		lt1.push_back(4);

		lt1.print();
	}

也是没有任何问题的

7,_list_iterator

迭代器的框架和初始化


	template<class T,class ref,class ptr>
	struct _list_iterator
	{
		typedef list_node<T> node;
		typedef _list_iterator<T,ref,ptr> sefl;
		node* _node;

		_list_iterator(node* n)
			:_node(n)
		{}
    }

有人会好奇,为什么模板里面有三个参数,现在先不急下面会进行分晓的;

指向结点的迭代器嘛,底层类型就是指针;

初始化也是一样,传来什么就是什么;

8,operator*()

迭代器解引用取值

		ref operator*()
		{
			return _node->data;
		}

ref 其实就是 T&;

9,begin()

找头结点

		iterator begin()
		{
			return iterator(_head->_next);
		}

直接返回构造完后的结果;

10,end()

最后一个结点的下一个位置

		iterator end()
		{
			return iterator(_head);
		}

然后我们就可以试一下迭代器打印了;

	void print_list(const list<int>& lt)
	{
		list<int>::const_iterator it = lt.begin();
		while (it != lt.end())
		{
			cout << *it << " ";
			++it;
		}
		cout << endl;
	}

	void test1()
	{
		wxd::list<int> lt1;
		lt1.push_back(1);
		lt1.push_back(2);
		lt1.push_back(3);
		lt1.push_back(4);

		print_list(lt1);
	}

11,operator->()

迭代器箭头指向取值

		ptr operator->()
		{
			return &_node->data;
		}

返回的是 data 的地址,ptr 是 T* ;

12,operator++()

迭代器前置++

		sefl& operator++()
		{
			_node = _node->_next;
			return *this;
		}

sefl 是  _list_iterator<T,ref,ptr>;

13,operator++(int)

迭代器后置++

		sefl operator++(int)
		{
			sefl tmp(*this);
			_node = _node->_next;
			return tmp;
		}

返回的是之前的值,但其实已经改变了;

14,operator--()

迭代器前置 - -

		sefl& operator--()
		{
			_node = _node->_prev;
			return *this;
		}

15,operator--(int)

迭代器后置 - -

		sefl operator--(int)
		{
			sefl tmp(*this);
			_node = _node->_prev;
			return tmp;
		}

16,operator==(const sefl& s)

迭代器判断相等

		bool  operator==(const sefl& s)
		{
			return _node == s._node;
		}

判断迭代器是否相等比较 _node 就可以了;

17,operator!=(const sefl& s)

判断是否不相等

		bool operator!=(const sefl& s)
		{
			return _node != s._node;
		}

18,_list_const_iterator

然后这个是 const 迭代器版本的,这里我就不一个一个写了;

template<class T>
	struct _list_const_iterator
	{
		typedef list_node<T> node;
		typedef _list_const_iterator<T> sefl;
		node* _node;

		_list_const_iterator(node* n)
			:_node(n)
		{}

		const T& operator*()
		{
			return _node->data;
		}

		sefl& operator++()
		{
			_node = _node->_next;
			return *this;
		}

		sefl operator++(int)
		{
			sefl tmp(*this);
			_node = _node->_next;
			return tmp;
		}

		sefl& operator--()
		{
			_node = _node->_prev;
			return *this;
		}

		sefl operator--(int)
		{
			sefl tmp(*this);
			_node = _node->_prev;
			return tmp;
		}

		bool  operator==(const sefl& s)
		{
			return _node == s._node;
		}

		bool operator!=(const sefl& s)
		{
			return _node != s._node;
		}

	};

其实吧,_list_const_iterator 跟 _list_iterator 就是内部函数参数的返回值不同罢了,我们可以用模板参数来实例化,这样就不用写两个迭代器了;

template <class T>
	class list
	{
		typedef list_node<T> node;
	public:
		typedef _list_iterator<T, T&, T*> iterator;
		typedef _list_iterator<T, const T&, const T*> const_iterator;

list 下面这样操作就可以了,普通迭代器模板一个版本,const 迭代器模板内的参数加上const就可以了,等调用的时候编译器会自动匹配的;

19,list(iterator first, iterator last)

迭代器区间构造

		template<class iterator>
		list(iterator first, iterator last)
		{
			empty_init();
			while (first != last)
			{
				push_back(*first);
				++first;
			}
		}

20,begin()const

const 版本取头结点

		const_iterator begin()const
		{
			return const_iterator(_head->_next);
		}

21,end()const

const 版本取尾结点的下一个位置

		const_iterator end()const
		{
			return const_iterator(_head);
		}

22,list(const list<T>& lt)

拷贝构造

		void swap(list<T>& tmp)
		{
			std::swap(_head, tmp._head);
		}

		list(const list<T>& lt)
		{
			empty_init();
			list<T> tmp(lt.begin(), lt.end());
			swap(tmp);
		}

先把要拷贝的区间信息构造另一个 list ,然后再与 this 指针的 _head哨兵位头结点进行交换即可;

23,operator=(list<T> lt)

赋值

		list<T>& operator=(list<T> lt)
		{
			swap(lt);
			return *this;
		}

24,insert(iterator pos, const T& x)

插入

		void insert(iterator pos, const T& x)
		{
			node* cur = pos._node;
			node* prev = cur->_prev;

			node* newnode = new node(x);
			
			prev->_next = newnode;
			newnode->_prev = prev;
			newnode->_next = cur;
			cur->_prev = newnode;
		}

定义前一个结点,和本身的结点,然后再进行插入即可;

	void test3()
	{
		wxd::list<int> lt1;
		lt1.push_back(1);
		lt1.push_back(2);
		lt1.push_back(3);
		lt1.push_back(4);

		auto pos = find(lt1.begin(), lt1.end(), 3);
		lt1.insert(pos, 9);
		print_list(lt1);
	}

25,erase(iterator pos)

擦除

		iterator erase(iterator pos)
		{
			assert(pos != end());

			node* next = pos._node->_next;
			node* tail = pos._node->_prev;

			tail->_next = next;
			next->_prev = tail;

			delete pos._node;
			return iterator(next);
		}

先断言一下,哨兵位结点是不能擦除的;

然后找到前一个结点,后一个结点,在进行互相绑定;

在释放要删除的空间;

26,clear()

清除

		void clear()
		{
			auto it = begin();
			while (it != end())
			{
				it=erase(it);
			}
		}

27,~list()

析构函数

		~list()
		{
			clear();
			delete _head;
			_head = nullptr;
		}

先清空结点,然后再是否哨兵位头结点置空即可;

28,push_front(const T& x)

头插

		void push_front(const T& x)
		{
			insert(begin(), x);
		}

直接用 insert 插入更加方便;

29,pop_front()

头删

		void pop_front()
		{
			erase(begin());
		}

30,pop_back()

尾删

		void pop_back()
		{
			erase(_head->_prev);
		}

31,源代码

#define _CRT_SECURE_NO_WARNINGS 1
#include<iostream>
#include<assert.h>
using namespace std;
	template<class T>
	struct list_node
	{
		list_node<T>* _next;
		list_node<T>* _prev;
		T data;

		list_node(const T& x = T())
			:_next(nullptr)
			, _prev(nullptr)
			, data(x)
		{}
	};

	template<class T,class ref,class ptr>
	struct _list_iterator
	{
		typedef list_node<T> node;
		typedef _list_iterator<T,ref,ptr> sefl;
		node* _node;

		_list_iterator(node* n)
			:_node(n)
		{}

		ref operator*()
		{
			return _node->data;
		}

		ptr operator->()
		{
			return &_node->data;
		}

		sefl& operator++()
		{
			_node = _node->_next;
			return *this;
		}

		sefl operator++(int)
		{
			sefl tmp(*this);
			_node = _node->_next;
			return tmp;
		}

		sefl& operator--()
		{
			_node = _node->_prev;
			return *this;
		}

		sefl operator--(int)
		{
			sefl tmp(*this);
			_node = _node->_prev;
			return tmp;
		}

		bool  operator==(const sefl& s)
		{
			return _node == s._node;
		}

		bool operator!=(const sefl& s)
		{
			return _node != s._node;
		}

	};

	template <class T>
	class list
	{
		typedef list_node<T> node;
	public:
		typedef _list_iterator<T, T&, T*> iterator;
		typedef _list_iterator<T, const T&, const T*> const_iterator;

		void empty_init()
		{
			_head = new node;
			_head->_next = _head;
			_head->_prev = _head;
		}

		list()
		{
			empty_init();
		}

		template<class iterator>
		list(iterator first, iterator last)
		{
			empty_init();
			while (first != last)
			{
				push_back(*first);
				++first;
			}
		}

		void push_back(const T& x)
		{
			node* tail = _head->_prev;
			node* newnode = new node(x);

			tail->_next = newnode;
			newnode->_prev = tail;
			newnode->_next = _head;
			_head->_prev = newnode;
		}

		iterator begin()
		{
			return iterator(_head->_next);
		}

		const_iterator begin()const
		{
			return const_iterator(_head->_next);
		}

		iterator end()
		{
			return iterator(_head);
		}

		const_iterator end()const
		{
			return const_iterator(_head);
		}

		void swap(list<T>& tmp)
		{
			std::swap(_head, tmp._head);
		}

		list(const list<T>& lt)
		{
			empty_init();
			list<T> tmp(lt.begin(), lt.end());
			swap(tmp);
		}

		list<T>& operator=(list<T> lt)
		{
			swap(lt);
			return *this;
		}


		void insert(iterator pos, const T& x)
		{
			node* cur = pos._node;
			node* prev = cur->_prev;

			node* newnode = new node(x);
			
			prev->_next = newnode;
			newnode->_prev = prev;
			newnode->_next = cur;
			cur->_prev = newnode;
		}

		iterator erase(iterator pos)
		{
			assert(pos != end());

			node* next = pos._node->_next;
			node* tail = pos._node->_prev;

			tail->_next = next;
			next->_prev = tail;

			delete pos._node;
			return iterator(next);
		}

		~list()
		{
			clear();
			delete _head;
			_head = nullptr;
		}

		void clear()
		{
			auto it = begin();
			while (it != end())
			{
				it=erase(it);
			}
		}

		void push_front(const T& x)
		{
			insert(begin(), x);
		}

		void pop_back()
		{
			erase(_head->_prev);
		}

		void pop_front()
		{
			erase(begin());
		}

		void print()
		{
			node* cur = _head->_next;
			while (cur != _head)
			{
				cout << cur->data << " ";
				cur = cur->_next;
			}
		}


	private:
		node* _head;
	};

32,总结

我们就先搞一个大概的,其中还有很多分支,比如我们写的是擦除某个数据,其实也可以擦除某个范围,这些就靠大家去摸索,查阅文档了;

list 类的实现就到这里了;

加油!

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Qt QComboBox组合框控件

文章目录 1 属性和方法1.1 文本1.2 图标1.3 插入和删除1.4 信号和槽 2 实例2.1 布局2.2 代码实现 Qt中的组合框是集按钮和下拉列表体的控件&#xff0c;&#xff0c;它占用的屏幕空间很小&#xff0c;对应的类是QComboBox 1 属性和方法 QComboBox有很多属性&#xff0c;完整的…

【福利】百度内容审核平台实战

文章目录 前言功能概述产品价格快速入门&#xff08;账号登录及资源领取、在线验证、编写示例程序&#xff09;实战演示1、首先创建一个应用2、引入百度的SDK3、测试用例百度内容审核-文本 200QPS百度内容审核-图像 50QPS 写在最后 前言 百度内容审核平台主要针对图像、文本、…

RT-Thread 中断管理

中断管理 什么是中断&#xff1f;简单的解释就是系统正在处理某一个正常事件&#xff0c;忽然被另一个需要马上处理的紧急事件打断&#xff0c;系统转而处理这个紧急事件&#xff0c;待处理完毕&#xff0c;再恢复运行刚才被打断的事件。 生活中&#xff0c;我们经常会遇到这…

开源C语言库Melon:数据恢复算法

本文讲述开源C语言库Melon中的里德所罗门纠错码的使用。 关于 Melon 库&#xff0c;这是一个开源的 C 语言库&#xff0c;它具有&#xff1a;开箱即用、无第三方依赖、安装部署简单、中英文文档齐全等优势。 Github repo 简介 里德所罗门编码是一种纠错码技术&#xff0c;…

C++ λ表达式

λ表达式提供了函数对象的另一种编程机制。 在 C 11 和更高版本中&#xff0c;Lambda 表达式&#xff08;通常称为 Lambda&#xff09;是一种在被调用的位置或作为参数传递给函数的位置定义匿名函数对象&#xff08;闭包&#xff09;的简便方法。 Lambda 通常用于封装传递给算法…

项目与工程的关系,是一个项目包含若干个工程还是一个工程包含若干个项目?

在项目管理和工程管理的领域里&#xff0c;项目&#xff08;Project&#xff09;和工程&#xff08;Engineering&#xff09;通常有不同的定义和关系&#xff0c;这取决于具体的行业和应用背景。但一般来说&#xff0c;项目和工程之间的关系可以这样理解&#xff1a; 项目包含工…

CMake入门教程【高级篇】管理MSVC编译器警告

😈「CSDN主页」:传送门 😈「Bilibil首页」:传送门 😈「动动你的小手」:点赞👍收藏⭐️评论📝 文章目录 1.什么是MSVC?2.常用的屏蔽警告3.MSVC所有警告4.target_compile_options用法5.如何在CMake中消除MSVC的警告?6.屏蔽警告编写技巧

docker一键安装

1.把docker_compose_install文件夹放在任意路径&#xff1b; 2.chmod -R 777 install.sh 3.执行./install.sh 兼容&#xff1a;CentOS7.6、麒麟V10服务器版、统信UOS等操作系统。 下载地址&#xff08;本人上传&#xff0c;免积分下载&#xff09;&#xff1a;https://downlo…

PPT模板,免费下载

找PPT模板、素材&#xff0c;就上这几个网站&#xff0c;免费下载。 1、菜鸟图库 https://www.sucai999.com/search/ppt/0_0_0_1.html?vNTYxMjky 菜鸟图库素材非常齐全&#xff0c;设计、办公、图片、视频等素材这里都能找到&#xff0c;PPT模板数量很可观&#xff0c;模板样…

每日一题——LeetCode1154.一年中的第几天

方法一 列举法&#xff1a; 用一个数组把每个月份的天数都列举出来 判断闰年&#xff0c;是闰年2月份有29天 循环对当前月份之前的月份天数求和 加上当天月份的天数 var dayOfYear function(date) {let year date.slice(0, 4);let month date.slice(5, 7);let day dat…

SpringBoot用MultipartFile.transferTo传递相对路径的问题

问题描述&#xff1a; 打算给自己的项目添加一个上传文件保存功能&#xff0c;于是我使用MultipartFile.transferTo()来完成这个功能&#xff0c;由于我的项目要部署到服务器&#xff0c;所以我使用了相对路径把上传的文件保存到当前项目的工作目录下&#xff0c;但是报错了&am…

基于SpringBoot的教学管理系统

文章目录 项目介绍主要功能截图&#xff1a;部分代码展示设计总结项目获取方式 &#x1f345; 作者主页&#xff1a;超级无敌暴龙战士塔塔开 &#x1f345; 简介&#xff1a;Java领域优质创作者&#x1f3c6;、 简历模板、学习资料、面试题库【关注我&#xff0c;都给你】 &…

黑马程序员——2022版软件测试——乞丐版——day01

目录&#xff1a; 测试介绍 什么是软件测试&#xff1f;测试主流技能主流方向建议测试常用分类 分类阶段划分代码可见度划分总结模型 角度质量模型测试流程 需求分析&#xff08;评审&#xff09;测试计划用例设计用例执行缺陷管理测试报告测试用例 用例的作用用例模板八大要素…

Camunda Spin

Spin 常用于在脚本中解析json或者xml使用&#xff0c;S(variable) 表示构造成Spin对象&#xff0c;通过prop(“属性名”)获取属性值&#xff0c;通过stringValue()、numberValue()、boolValue() 等对类型转换。 repositoryService.createDeployment().name("消息事件流程&…

Fluids —— Fluid-object collisions

对移动碰撞体的精确速度&#xff0c;通常对模拟是非常重要的&#xff0c;尤其是FLIP&#xff1b;不正常的碰撞速度&#xff0c;可能会缺乏动态的飞溅或泄漏&#xff1b; SOP流体通过FLIP Collide SOP节点来处理碰撞和交互&#xff1b;碰撞对象可以是静态达到&#xff0c;移动的…

Druid 分析jpa批量插入

Druid是阿里巴巴开发的号称为监控而生的数据库连接池&#xff0c;在功能、性能、扩展性方面&#xff0c;都超过其他数据库连接池&#xff0c;包括DBCP、C3P0、BoneCP、Proxool、JBoss DataSource等&#xff0c;秒杀一切。 Druid 可以很好的监控 DB 池连接和 SQL 的执行情况&am…

浅析NVMe key per IO加密技术-1

一、Key per IO功能介绍 在当前的数据中心环境中&#xff0c;数据加密通常采用存储设备内部生成和管理的加密密钥与加密数据紧密耦合的方式进行。TCG Opal定义了一套针对自加密硬盘&#xff08;SED, Self-Encrypting Drives&#xff09;的标准化安全子系统类&#xff08;SSC, …

Go语言学习笔记(三)

教程&#xff1a;文档 - Go 编程语言 (studygolang.com) 调用模块代码 在call-module-code需要注意&#xff0c;需要在hello目录下操作 go mod edit -replace example.com/greetings../greetings 这是一个在Go项目的模块管理中的命令。在Go的模块管理工具&#xff08;go mod&…

K8s Pod详解

1.Pod结构 每个Pod中都可以包含一个或者多个容器&#xff0c;这些容器可以分为两类&#xff1a; 用户程序所在的容器&#xff0c;数量可多可少 Pause容器&#xff0c;这是每个Pod都会有的一个根容器&#xff0c;它的作用有两个&#xff1a; 可以以它为依据&#xff0c;评估整个…

基于java的SSM框架实现在线投稿网站系统项目【项目源码+论文说明】计算机毕业设计

基于java的SSM框架Vue实现在线投稿网站系统演示 摘要 随着计算机技术的飞速发展&#xff0c;稿件也已进入信息化时代。为了使稿件管理更高效、更科学&#xff0c;决定开发投稿审稿系统。 本文采用自顶向下的结构化的系统分析方法&#xff0c;阐述了一个功能全面的投稿审稿系统…