【C++初阶】list的模拟实现

news2025/1/11 20:01:09

文章目录

  • list的介绍
  • list的模拟实现
    • 成员变量
    • Member functions
      • constructor
      • destructor
      • operator=
    • Iterators
      • 正向迭代器
      • 反向迭代器
      • begin
      • end
      • rbegin
      • rend
    • Modifiers
      • push_front
      • pop_front
      • push_back
      • pop_back
      • insert
      • erase
      • clear
  • 完整版代码
    • list.h
    • reverse_iterator.h
    • test.cpp

list的介绍

list是STL库里面的一个重要容器,它分为一下几个部分
Member functions
在这里插入图片描述

Iterators
在这里插入图片描述

Capacity
在这里插入图片描述

Element access
在这里插入图片描述

Modifiers
在这里插入图片描述

Operations
在这里插入图片描述

Observers
在这里插入图片描述

Non-member function overloads
在这里插入图片描述

list的模拟实现

我们重点实现list的下面几个部分,其余部分有的还没学到,有的比较简单也不是特别常用大家可以自己去完善。
我们实现的链表结构为带头双向循环链表
在这里插入图片描述
_head为哨兵位头节点,是不存储数据的

成员变量

在这里插入图片描述

在这里插入图片描述
_head为哨兵位头节点
节点指针
在这里插入图片描述
节点
_prev记录前一个节点的位置
_next记录下一个节点的位置

Member functions

constructor

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
_head为哨兵位头节点
在这里插入图片描述
_head为哨兵位头节点
在这里插入图片描述
_head为哨兵位头节点
在这里插入图片描述
_head为哨兵位头节点
在这里插入图片描述
_head为哨兵位头节点

destructor

在这里插入图片描述

在这里插入图片描述
_head为哨兵位头节点

operator=

在这里插入图片描述
_head为哨兵位头节点

Iterators

链表的正向迭代器和反向迭代器和之前学的string、vector不同,因为它们的物理空间连续可以++、–随机访问它们的各个位置所以它们的迭代器为原生指针,而list的物理空间不连续,不能++、–的随机访问任意位置,所以我们要对list的迭代器进行封装。

正向迭代器

在这里插入图片描述

在这里插入图片描述

反向迭代器

在这里插入图片描述

在这里插入图片描述

begin

在这里插入图片描述
_head为哨兵位头节点
在这里插入图片描述
_head为哨兵位头节点
const迭代器和普通迭代器的区别在于只可读,不可写

end

在这里插入图片描述
_head为哨兵位头节点
在这里插入图片描述
_head为哨兵位头节点
const迭代器和普通迭代器的区别在于只可读,不可写

rbegin

在这里插入图片描述
在这里插入图片描述
const迭代器和普通迭代器的区别在于只可读,不可写

rend

在这里插入图片描述
在这里插入图片描述
const迭代器和普通迭代器的区别在于只可读,不可写

Modifiers

push_front

在这里插入图片描述
复用insert,因为begin()是第一个数据的位置,头插就是要在第一个数据位置的前面插入数据。

pop_front

在这里插入图片描述
复用erase,因为begin()是第一个数据的位置,头删是要删除第一个数据的位置。

push_back

在这里插入图片描述
复用insert,因为end()是头节点的位置,尾插就是要在头节点前插入数据。(注释部分为不复用的实现方法)

pop_back

在这里插入图片描述
复用erase,因为end()是头节点的位置,尾删是要删除头节点的前一个位置,所以要–

insert

在这里插入图片描述
在这里插入图片描述

erase

在这里插入图片描述
在这里插入图片描述

clear

在这里插入图片描述
在这里插入图片描述

完整版代码

list.h

#pragma once
#include"reverse_iterator.h"
namespace lzf
{
	
	template<class T>
	struct ListNode
	{
	public:
		T _data;
		ListNode<T>* _prev;
		ListNode<T>* _next;
		ListNode(const T& val = T())
			:_data(val)
			, _prev(nullptr)
			,_next(nullptr)
		{}
		
	};

	template<class T,class Ref,class Ptr>
	struct list_iterator
	{
		typedef list_iterator<T, Ref, Ptr> self;
		typedef ListNode<T> Node;
		Node* _node;
		list_iterator(Node* x)
			:_node(x)
		{}
		Ref operator*()
		{
			return _node->_data;
		}
		Ptr operator->()
		{
			return &_node->_data;
		}
		self& operator++()
		{
			_node = _node->_next;

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

			return temp;
		}
		self& operator--()
		{
			_node = _node->_prev;

			return *this;
		}
		self operator--(int)
		{
			self temp(_node);
			_node = _node->_prev;

			return temp;
		}
		/*bool operator!=(self& it)
		{
			return _node != it._node;
		}
		bool operator==(self& it)
		{
			return _node == it._node;
		}*/
		/*bool operator!=(self it)
		{
			return _node != it._node;
		}
		bool operator==(self it)
		{
			return _node == it._node;
		}*/
		bool operator!=(const self& it)const
		{
			return _node != it._node;
		}
		bool operator==(const self& it)const
		{
			return _node == it._node;
		}
	};
	

	template<class T>
	class list
	{
		typedef ListNode<T> Node;
	public:
		typedef list_iterator<T, T&, T*> iterator;
		typedef list_iterator<T, const T&, const T*> const_iterator;

		typedef reverse_iterator<iterator, const T&, const T*> const_reverse_iterator;
		typedef reverse_iterator<iterator, T&, T*> reverse_iterator;
		reverse_iterator rbegin()
		{
			return reverse_iterator(end());
		}
		reverse_iterator rend()
		{
			return reverse_iterator(begin());
		}
		const_reverse_iterator rbegin()const
		{
			return const_reverse_iterator(end());
		}
		const_reverse_iterator rend()const
		{
			return const_reverse_iterator(begin());
		}
		iterator begin()
		{
			return iterator(_head->_next);
		}
		iterator end()
		{
			return iterator(_head);
		}
		const_iterator begin()const
		{
			return const_iterator(_head->_next);
		}
		const_iterator end()const
		{
			return const_iterator(_head);
		}
		
		list()
		{
			_head = new Node;
			_head->_next = _head;
			_head->_prev = _head;
		}
		list(int n,const T& val = T())
		{
			_head = new Node;
			_head->_next = _head;
			_head->_prev = _head;
			for (int i = 0; i < n; i++)
			{
				push_back(val);
			}
		}
		list(size_t n, const T& val = T())
		{
			_head = new Node;
			_head->_next = _head;
			_head->_prev = _head;
			for (size_t i = 0; i < n; i++)
			{
				push_back(val);
			}
		}
		template<class Inputiterator>
		list(Inputiterator first, Inputiterator last)
		{
			_head = new Node;
			_head->_next = _head;
			_head->_prev = _head;
			while (first != last)
			{
				push_back(*first);
				++first;
			}
		}
		list(const list<T>& lt)
		{
			_head = new Node;
			_head->_next = _head;
			_head->_prev = _head;
			/*const_iterator it = lt.begin();
			while (it != lt.end())
			{
				push_back(*it);
				++it;
			}*/
			list<T> temp(lt.begin(), lt.end());
			std::swap(_head, temp._head);
		}
		list<T>& operator=(list<T> lt)
		{
			/*_head = new Node;
			_head->_next = _head;
			_head->_prev = _head;*/
			std::swap(_head, lt._head);
			return *this;
		}
		void push_back(const T& x)
		{
			/*Node* newnode = new Node(x);
			Node* tail = _head->_prev;
			tail->_next = newnode;
			newnode->_prev = tail;

			newnode->_next = _head;
			_head->_prev = newnode;*/
			insert(end(), x);
		}
		void pop_back()
		{
			erase(--end());
		}
		void push_front(const T& x)
		{
			insert(begin(),x);
		}
		void pop_front()
		{
			erase(begin());
		}
		iterator insert(iterator pos, const T& x)
		{
			Node* newnode = new Node(x);
			Node* prev = pos._node->_prev;
			//Node* next = pos._node->_next;

			prev->_next = newnode;
			newnode->_prev = prev;

			pos._node->_prev = newnode;
			newnode->_next = pos._node;

			return iterator(newnode);
		}
		iterator erase(iterator pos)
		{
			assert(pos != end());
			Node* prev = pos._node->_prev;
			Node* next = pos._node->_next;

			delete pos._node;

			prev->_next = next;
			next->_prev = prev;


			return iterator(next);
		}
		void clear()
		{
			iterator it = begin();
			while (it != end())
			{
				it = erase(it);
			}
		}
		~list()
		{
			clear();
			delete _head;
			_head = nullptr;
		}
	private:
		Node* _head;
	};
	void test_list1()
	{
		list<int> lt;
		lt.push_back(1);
		lt.push_back(2);
		lt.push_back(3);
		lt.push_back(4);
		list<int>::iterator it = lt.begin();
		while (it != lt.end())
		{
			cout << *it << " ";
			it++;
		}
		//list<int> 
	}
	void test_list2()
	{
		list<int> lt;
		lt.push_back(1);
		lt.push_back(2);
		lt.push_back(3);
		lt.push_back(4);
		//list<int> lt2(5,1);
		//list<int> lt2(lt.begin(), lt.end());
		//list<int> lt2(lt);
		list<int> lt2;
		lt2.push_back(7);
		lt2.push_back(8);
		lt2.push_back(9);
		lt2.push_back(10);

		//lt.operator=(lt2);
		//list<int>::iterator it = lt.begin();
		/*while (it != lt.end())
		{
			cout << *it << " ";
			it++;
		}*/
		//list<int> 
	}
	void test_list3()
	{
		list<int> lt;
		lt.push_back(1);
		lt.push_back(2);
		lt.push_back(3);
		lt.push_back(4);
		lt.insert(lt.begin(), 1);
		lt.insert(lt.begin(), 2);
		lt.insert(lt.begin(), 3);
		lt.insert(lt.begin(), 4);
		lt.insert(lt.end(), 4);
		lt.insert(lt.end(), 3);
		lt.insert(lt.end(), 2);
		lt.insert(lt.end(), 1);
		list<int>::iterator it = lt.begin();
		while (it != lt.end())
		{
			cout << *it << " ";
			it++;
		}
		cout << endl;
		lt.erase(lt.begin());
		lt.erase(lt.begin());
		lt.erase(lt.begin());
		lt.erase(lt.begin());
		lt.erase(--lt.end());
		lt.erase(--lt.end());
		lt.erase(--lt.end());
		lt.erase(--lt.end());
		it = lt.begin();
		while (it != lt.end())
		{
			cout << *it << " ";
			it++;
		}

	}
	void test_list4()
	{
		list<int> lt;
		lt.push_back(1);
		lt.push_back(2);
		lt.push_back(3);
		lt.push_back(4);
		lt.clear();
		list<int>::iterator it = lt.begin();
		while (it != lt.end())
		{
			cout << *it << " ";
			it++;
		}
		lt.push_back(1);
		lt.push_back(2);
		lt.push_back(3);
		lt.push_back(4);
		it = lt.begin();
		while (it != lt.end())
		{
			cout << *it << " ";
			it++;
		}
	}
	void test_list5()
	{
		list<int> lt;
		lt.push_back(1);
		lt.push_back(2);
		lt.push_back(3);
		lt.push_back(4);
		list<int>::reverse_iterator it = lt.rbegin();
		while (it != lt.rend())
		{
			cout << *it << " ";
			//++it;
			it++;
		}
	}
	void test_list6()
	{
		list<int> lt;
		lt.push_back(1);
		lt.push_back(2);
		lt.push_back(3);
		lt.push_back(4);
		list<int> lt1;
		list<int> lt2;
		lt2 = lt1 = lt;
		list<int>::reverse_iterator it = lt2.rbegin();
		while (it != lt2.rend())
		{
			cout << *it << " ";
			//++it;
			it++;
		}
	}
}

reverse_iterator.h

#pragma once
namespace lzf
{
	template<class iterator, class Ref, class Ptr>
	class reverse_iterator
	{
	public:
		typedef reverse_iterator<iterator, Ref, Ptr> self;
		reverse_iterator(iterator it)
			:_it(it)
		{}
		Ref operator*()
		{
			iterator temp(_it);
			return *--temp;
		}
		Ptr operator->()
		{
			iterator temp(_it);
			return &(*--temp);
		}
		self& operator++()
		{
			--_it;
			return *this;
		}
		self& operator--()
		{
			++_it;
			return *this;
		}
		self operator++(int)
		{
			self temp(_it);
			--_it;
			return temp;
		}
		self operator--(int)
		{
			self temp(_it);
			++_it;
			return temp;
		}
		bool operator!=(const self& it)const
		{

			return _it != it._it;
		}
		bool operator==(const self& it)const
		{

			return _it == it._it;
		}
	private:
		iterator _it;
	};
}

test.cpp

#include<iostream>
#include<assert.h>
using namespace std;
#include"list.h"
int f()
{
	int a = 10;
	//double& b = a;
	return a;
}
int main()
{
	//int& b = f();
	lzf::test_list6();
	return 0;
}

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

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

相关文章

Selenium自动化测试环境搭建及问题

一、前言 目标&#xff1a;环境搭建及测试 主要问题&#xff1a;浏览器闪退&#xff0c;路径不明确&#xff0c;语法弃用 二、环境搭建&#xff08;自带python3.9以上&#xff09; 2.1.1、下载谷歌浏览器驱动 https://registry.npmmirror.com/binary.html?pathchromedriver/&…

数据结构---A星寻路算法

A星寻路算法第一步第二步第三步第四步JAVA实现用于寻找有效路径的算法。定义俩个集合 OpenList&#xff1a;可到达的格子 CloseList&#xff1a;已到达的格子 每一个格子都具有F、G、H这3个属性 G&#xff1a;从起点走到当前格子的成本&#xff0c;也就是已经花费了多少步。H&a…

生产级部署 Python 脚本,崩溃可自启

今天介绍一个生产级的流程管理工具 PM2&#xff0c;通常我们说到 PM2 的时候&#xff0c;都是在说如何部署 Node.js 程序&#xff0c;但是实际上 PM2 很强大&#xff0c;不仅仅可以用来管理 Node.js&#xff0c;它还可以用来管理 Python、PHP、Ruby、perl 等等。 这里就以 Pyt…

看了下华为工资,我不加班了

周五快下班&#xff0c;我本来是想继续好好上班的。那时候是晚上8点左右&#xff0c;跟我一个华为的朋友聊天&#xff0c;聊完之后&#xff0c;我气得把电脑合上&#xff0c;拿上花了7万巨款买的车钥匙&#xff0c;头也不回的走到电梯口&#xff0c;按下了下楼的电梯按钮。-事情…

项目集成MybatisPlus

目录 1.MyBatisPlus介绍 1.1.为什么需要 1.2.MyBatisPlus简介 1.3.MybatisPlus特点 2.项目集成MybatisPlus 2.1.导入依赖 2.2.创建配置 2.3.代码生成主类GenteratorCode 2.4.Controller&#xff0c;Query模板 2.5.生成代码 2.6最终效果 1.MyBatisPlus介绍 1.1.为什么…

SQL优化篇:如何成为一位写优质SQL语句的绝顶高手

(Structured Query Language)标准结构化查询语言简称SQL&#xff0c;编写SQL语句是每位后端开发日常职责中&#xff0c;接触最多的一项工作&#xff0c;SQL是关系型数据库诞生的产物&#xff0c;无论是什么数据库&#xff0c;MySQL、Oracle、SQL Server、DB2、PgSQL....&#x…

JSP ssh网上商品拍卖系统myeclipse开发mysql数据库MVC模式java编程计算机网页设计

一、源码特点 JSP ssh网上商品拍卖系统是一套完善的web设计系统&#xff08;系统采用ssh框架进行设计开发&#xff09;&#xff0c;对理解JSP java编程开发语言有帮助&#xff0c;系统具有完整的源代码和数据库&#xff0c;系统主要采用B/S 模式开发。开发环境为TOMCAT7.0…

个人收款的实现方案

最近出于个人需要折腾了一套微信免签支付的个人系统,下面介绍一下整体的一个实现方案 由于微信本身不提供免签的支付api,因此个人若想要实现免签支付,那么只能通过监听微信收款通知来实现了,因此整个系统的实现部分分为 收款服务端和监听客户端来实现 收款服务端实现 整个…

【nowcoder】笔试强训Day5

目录 一、单选题 二、多选题 三、编程题 3.1统计回文 3.2连续最大和 一、单选题 1.下面的程序 编译运行后&#xff0c;在屏幕上显示的结果是&#xff08;&#xff09; public class Test {public static void main(String args[]) {int x, y;x 5 >> 2;y x >&…

docker入门以及常见的命令

目录 1. 什么是docker 2. docker的核心组件 3. docker的安装 3.1 安装的先决条件 3.2.1 ubuntu安装docker 3.2.2 CentOS安装docker 3.3 配置镜像加速器 4. 镜像常用操作 4.1 搜索镜像 4.2 镜像下载 4.3 查看宿主机中的镜像 4.3 删除镜像 5. 容器常用命令 5.1 运行…

#Z1243. 完美数

一&#xff0c;题目描述 一个数是完美的&#xff0c;仅当它等于它的因数中比它小的所有数之和。 例如&#xff1a;28124714&#xff0c;所以 28 是完美的。 由此我们可以定义一个数的不完美值 F(N)&#xff0c;代表 N和比 N 小的所有 N 的因数之和的差的绝对值。 例如&…

c# 服务

是什么 Microsoft Windows 服务&#xff08;过去称为 NT 服务&#xff09;允许用户创建可在其自身的 Windows 会话中长时间运行的可执行应用程序。 这些服务可在计算机启动时自动启动&#xff0c;可以暂停和重启&#xff0c;并且不显示任何用户界面。 这些功能使服务非常适合在…

metrics-server监控主机资源

使用metrics-server实现主机资源监控获取metrics-server资源清单文件修改metrics-server资源清单文件部署metrics-server资源清单文件验证及授权获取metrics-server资源清单文件 直接使用命令来获取资源清单文件 wget https://github.com/kubernetes-sigs/metrics-server/rele…

车用DCDC双路输出电源模块规格书

产品名称&#xff1a;非隔离稳压双路输出车用电源模块 产品型号&#xff1a;LM36J40W2EH 典型特征值&#xff1a;24V/8V(3A)&5V(2A) u 产品概述 LM36J40W2EH模块电源是一款非隔离型双路输出的开关稳压器。它采用集成IC设计&#xff0c;使用一进两出的接线方式&#xff…

【Html圣诞树】2022年的圣诞节了,祝大家节日快乐。

目录 1. 效果展示2. 源代码1. 效果展示 是最近在互联网上很火的一个效果。 是带有背景音乐的,而且背景音乐支持选择,也支持自定义。 2. 源代码 美丽的圣诞树.html <!DOCTYPE html> <html lang="en"

从零学习 InfiniBand-network架构(十) —— IB协议中全局ID

从零学习 InfiniBand-network架构&#xff08;十&#xff09; —— IB协议中全局ID &#x1f508;声明&#xff1a; &#x1f603;博主主页&#xff1a;王_嘻嘻的CSDN主页 &#x1f511;未经作者允许&#xff0c;禁止转载 &#x1f6a9;本专题部分内容源于《InfiniBand-network…

设备指示灯开关状态识别检测系统 yolov5

设备指示灯开关状态识别检测系统是基于yolo网络深度学习模型&#xff0c;对现场画面进行实时监测识别。自动识别仪表示数或开关状态。我们使用YOLO(你只看一次)算法进行对象检测。YOLO是一个聪明的卷积神经网络(CNN)&#xff0c;用于实时进行目标检测。该算法将单个神经网络应用…

Python中使用zip函数的七重境界

1. 引言 Python中有一些内置函数&#xff0c;可以使我们的代码非常优雅。zip 函数就是其中之一&#xff0c;但是zip 函数的使用对于初学者来说不是很直观&#xff0c;有时容易出错。因此本文将从7个层次来由浅入深地来探讨强大的zip 函数的概念、用法和技巧。 闲话少说&#x…

用HTML写一个2023跨年动画代码(烟花+自定义文字+背景音乐+雪花+倒计时)

*2022年圣诞节到来啦&#xff0c;很高兴这次我们又能一起度过~ 文章目录一、前言二、跨年烟花三、效果展示五、HTML源码一、前言 时光荏苒&#xff0c;白驹过隙。 2022这一年又在忙碌中度过了&#xff0c;过去的一年&#xff0c;我们同努力&#xff0c;我们共欢笑&#xff0c;…

ContentProvider基础知识

ContentProvider基础知识 1.ContentProvider入门 1.简介 不同程序之间的数据交换&#xff0c;不同app之间的数据共享学会如何操作数据学会如何监听数据变化学会URI,URImatcher&#xff0c;ContentUris的如何使用2.基本使用 A应用&#xff0c;使用ContentProvider的子类进行暴漏…