当前位置: 首页 > article >正文

list的模拟实现详解

文章目录

  • list的模拟实现
    • list的迭代器
    • begin()和end()

list的模拟实现

#pragma once
#include<iostream>
#include<list>

using namespace std;

namespace wbc
{
	// 类模版
	template<class T>
	struct list_node // 链表的节点
	{
		T _data;
		list_node<T>* _next;
		list_node<T>* _prev;

		// 默认构造
		list_node(const T& val = T())
			:_data(val),
			_next(nullptr),
			_prev(nullptr)
		{

		}
	};

	template<class T>
	struct list_iterator
	{
		typedef list_node<T> Node;
		typedef list_iterator<T> Self;
		Node* _node;

		// 构造
		list_iterator(Node* node)
			:_node(node)
		{

		}

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

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

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

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

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

	template<class T>
	class list // 链表的结构
	{
		typedef list_node<T> Node;
	public:
		typedef list_iterator<T> iterator;

		iterator begin()
		{
			// 有名对象
			/*iterator it(_head->_next);
			return it;*/

			// 匿名对象
			//return iterator(_head->_next);


			return _head->_next;
			// 单参数的构造函数支持隐式类型的转化
		}

		// end是最后一个位置的下一个
		iterator end()
		{
			return _head;
		}

		// 构造
		list()
		{
			_head = new Node;
			_head->_next = _head;
			_head->_prev = _head;
			_size = 0;
		}

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

		// 尾插
		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;

			++_size;*/

			insert(end(), x);
		}
		

		// 插入,在pos之前插入一个x
		void insert(iterator pos,const T& x)
		{
			Node* cur = pos._node;
			Node* prev = cur->_prev;

            // prev newnode cur
			Node* newnode = new Node(x);
			prev->_next = newnode;
			cur->_prev = newnode;
			newnode->_next = cur;
			newnode->_prev = prev;

			++_size;
		}

		
		// 删除
		void erase(iterator pos)
		{
			// 不能删除哨兵位的头节点
			assert(pos != end());

			Node* cur = pos._node;
			Node* prev = cur->_prev;
			Node* next = cur->_next;

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

			delete cur;
			--_size;
		}

		// 尾删
		void pop_back()
		{
			erase(--end());
		}

		// 头删
		void pop_front()
		{
			erase(begin());
		}

		size_t size() const
		{
			return _size;
		}

		bool empty() const
		{
			return _size == 0;
		}
	private:
		Node* _head;
		size_t _size;
	};

	void test_list1()
	{
		list<int> ls;

		ls.push_back(1);
		ls.push_back(2);
		ls.push_back(3);
		ls.push_back(4);
		ls.push_back(5);


		list<int>::iterator it = ls.begin();
		while (it != ls.end())
		{
			cout << *it << endl;
			++it;
		}
		cout << endl;

		for (auto ch : ls)
		{
			cout << ch << " ";
		}
		cout << endl;
	}
}

list的迭代器

1.list迭代器重载的operator++不是原生指针,不能直接加到下一个位置,vector,string的迭代器是原生指针可以直接加到下一个位置
2.list的operator* 也需要重载不然* 之后是一个节点,++不能到下一个位置,实现的operator *解引用拿到的节点里的值

    template<class T>
	struct list_iterator
	{
		typedef list_node<T> Node;
		typedef list_iterator<T> Self;
		Node* _node;

		// 构造
		list_iterator(Node* node)
			:_node(node)
		{}

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

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

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

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

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

begin()和end()

        iterator begin()
		{
			// 有名对象
			/*iterator it(_head->_next);
			return it;*/

			// 匿名对象
			//return iterator(_head->_next);

            // 返回的是哨兵位节点的下一个位置
			return _head->_next;
			// 单参数的构造函数支持隐式类型的转化
		}

		// end是最后一个位置的下一个
		iterator end()
		{
			return _head;
		}

http://www.kler.cn/a/507002.html

相关文章:

  • LabVIEW 程序中的 R6025 错误
  • LLMs之RAG:《EdgeRAG: Online-Indexed RAG for Edge Devices》翻译与解读
  • 使用中间件自动化部署java应用
  • GCN详细介绍:原理、主要应用
  • 【Go】Go Gin框架初识(一)
  • (STM32笔记)十二、DMA的基础知识与用法 第二部分
  • 核心前端技术详解
  • Jupyter notebook中运行dos指令运行方法
  • Java进阶-在Ubuntu上部署SpringBoot应用
  • 微软开源AI Agent AutoGen 详解
  • Docker部署Spring Boot + Vue项目
  • ParcelFileDescriptor+PdfRenderer在Android渲染显示PDF文件
  • Spring Boot中使用AOP实现权限管理
  • Python 的时间处理模块 datetime 详解
  • 图论1-问题 B: 算法7-4,7-5:图的遍历——深度优先搜索
  • 博图 linucx vmware
  • css 实现自定义虚线
  • QT 通过QAxObject与本地应用程序读取Excel内容
  • 汽车故障码U100187 LIN1Communication time out 解析和处理方法
  • 【50个服务器常见端口】
  • 【Linux】sed编辑器二
  • 基于华为云车牌识别服务设计的停车场计费系统【华为开发者空间-鸿蒙】
  • ArcGIS模拟风场(流场)
  • 《AI与鸿蒙Next:建筑设计可视化的革新力量》
  • 使用AKTools本地部署AKShare财经数据接口库
  • 《零基础Go语言算法实战》【题目 4-12】找到给定集合的所有子集