C++:list使用以及模拟实现

这篇具有很好参考价值的文章主要介绍了C++:list使用以及模拟实现。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

list介绍

  1. list是一个类模板,加<类型>实例化才是具体的类
  2. list是可以在任意位置进行插入和删除的序列式容器。
  3. list的底层是双向循环链表结构,链表中每个元素存储在互不相关的独立节点中,在节点中通过指针指向其前一个元素和后一个元素。
  4. 与其他序列式容器相比,list最大的缺陷是不支持任意位置的随机访问,比如访问第六个节点,必须从已知节点迭代到该节点。

双向链表图解:
C++:list使用以及模拟实现,C++初阶,c++,list,开发语言,stl,笔记



list常用接口

1.构造

函数 功能
list (size_type n, const value_type& val = value_type()) 构造的list中包含n个值为val的节点
list() 构造空的list
list (const list& x) 拷贝构造
list (InputIterator first, InputIterator last) 迭代器区间初始化
(模板,可以传入其它容器的迭代器区间)

2.迭代器

函数 功能
begin()加end() 获取第一个数据位置的iterator/const_iterator,获取最后一个数据的下一个位置的iterator/const_iterator
rbegin()加rend() 反向迭代器,获取最后一个数据位置的reverse_iterator,获取第一个数据前一个位置的reverse_iterator

3.容量

函数 功能
size() 获取有效数据个数
empty() 判断是否为空(size为0为空,返回true)

4.访问数据

函数 功能
front() 取到头节点数据的引用
back() 返回尾节点数据的引用

5.增删查改

函数 功能
push_front
(const value_type& val)
头插数据val
push_back
(const value_type& val)
尾删数据val
pop_front() 头删
pop_back() 尾删
insert (iterator position, const value_type& val) 在position位置中插入值为val的元素
erase (iterator position) 删除position位置的元素
swap(list& x) 交换两个list
clear() 清空有效数据

6.迭代器失效

       迭代失效即迭代器所指向的节点的无效,即该节点被删除了。 因为list的底层结构为带头结点的双向循环链表,因此在list中进行插入时是不会导致list的迭代器失效的,只有在删除时才会失效,并且失效的只是指向被删除节点的迭代器,其他迭代器不会受到影响。

#include<iostream>
#include<list>
using namespace std;

//错误代码演示
int main()
{
	int array[] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 0 };
	list<int> l(array, array + sizeof(array) / sizeof(array[0]));
	auto it = l.begin();
	while (it != l.end())
	{
		// erase()函数执行后,it所指向的节点已被删除
		//因此it无效,在下一次使用it时,必须先给其赋值
		it = l.erase(it); //只需要去掉++it,这里修改成it = erase(it)即可
		++it;
	}
	return 0;
}



list模拟实现

1.迭代器的实现

      有别于vector,list的迭代器并不是一个原生的指针,因为使用者需要得到的是节点中的数据而不是整个节点,而且寻找下一个节点并不能通过指针简单的++,所以需要把迭代器单独封装成一个类,通过重载*等运算符来完成需求。文章来源地址https://www.toymoban.com/news/detail-666097.html

namespace MyList
{
	//节点设计成结构体,方便访问
	template<typename T>
	struct list_node
	{
		list_node(const T val = T())
			:_data(val)
			, _next(nullptr)
			, _prev(nullptr)
		{}
		T _data;
		list_node<T>* _next;
		list_node<T>* _prev;
	};

	//迭代器
	//这里设计模板参数除了迭代器,还有Ref(引用)和Ptr(指针)
	//这样设计是为了同时生成普通迭代器和const对象的迭代器
	//普通对象(可读可写):iterator<T, T&, T*>
	//const对象(可读不可写):const_iterator<T, const T&, const T*>
	template<typename T, typename Ref, typename Ptr>
	struct __list_iterator
	{
		typedef list_node<T> Node;
		typedef __list_iterator<T, Ref, Ptr> self; //要返回迭代器需要返回实例化对象,重命名一下

		Node* _node;

		__list_iterator(Node* p)
			:_node(p)
		{}

		self& operator++()
		{
			_node = _node->_next;
			return *this;
		}
		
		//后置++
		self operator++(int)
		{
			self tmp(*this);
			_node = _node->_next;
			return tmp;
		}

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

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

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

		//返回指针可以让自定义类型自行打印,访问成员
		//->操作符,比较特殊,it->_num转换出来其实是it.operator->()->_num
		Ptr operator->()
		{
			return &(_node->_data);
		}

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

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

	//反向迭代器
	//反向迭代器需要进行封装,其实就是复用普通迭代器,然后++和--操作反过来
	
	//普通对象(可读可写):Reverse_iterator<iterator,T&,T*>
	//const对象(可读不可写):Reverse_iterator<const_iterator,const T&,const T*>
	template<class Iterator, class Ref, class Ptr>
	struct Reverse_iterator
	{
		typedef Reverse_iterator<Iterator, Ref, Ptr> self;
		Iterator _it;

		//构造
		Reverse_iterator(Iterator it)
			:_it(it)
		{}


		self& operator++()
		{
			_it--;
			return *this;
		}


		self operator++(int)
		{
			self tmp(*this);
			_it--;
			return tmp;
		}


		self& operator--()
		{
			_it++;
			return *this;
		}


		self operator--(int)
		{
			self tmp(*this);
			_it++;
			return tmp;
		}


		Ref operator*()
		{
			return *_it;
		}


		Ptr operator->()
		{
			return _it;
		}


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


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

2.完整代码

#pragma once
#include<iostream>
using namespace std;

namespace MyList
{
	//节点设计成结构体,方便访问
	template<typename T>
	struct list_node
	{
		list_node(const T val = T())
			:_data(val)
			, _next(nullptr)
			, _prev(nullptr)
		{}
		T _data;
		list_node<T>* _next;
		list_node<T>* _prev;
	};

	//迭代器
	//这里设计模板参数除了迭代器,还有Ref(引用)和Ptr(指针)
	//这样设计是为了同时生成普通迭代器和const对象的迭代器
	//普通对象(可读可写):iterator<T, T&, T*>
	//const对象(可读不可写):const_iterator<T, const T&, const T*>
	template<typename T, typename Ref, typename Ptr>
	struct __list_iterator
	{
		typedef list_node<T> Node;
		typedef __list_iterator<T, Ref, Ptr> self; //要返回迭代器需要返回实例化对象,重命名一下

		Node* _node;

		__list_iterator(Node* p)
			:_node(p)
		{}

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

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

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

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

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

		//返回指针可以让自定义类型自行打印,访问成员
		//->操作符,比较特殊,it->_num转换出来其实是it.operator->()->_num
		Ptr operator->()
		{
			return &(_node->_data);
		}

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

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

	//反向迭代器
	//反向迭代器需要进行封装,其实就是复用普通迭代器,然后++和--操作反过来
	
	//普通对象(可读可写):Reverse_iterator<iterator,T&,T*>
	//const对象(可读不可写):Reverse_iterator<const_iterator,const T&,const T*>
	template<class Iterator, class Ref, class Ptr>
	struct Reverse_iterator
	{
		typedef Reverse_iterator<Iterator, Ref, Ptr> self;
		Iterator _it;


		Reverse_iterator(Iterator it)
			:_it(it)
		{}


		self& operator++()
		{
			_it--;
			return *this;
		}


		self operator++(int)
		{
			self tmp(*this);
			_it--;
			return tmp;
		}


		self& operator--()
		{
			_it++;
			return *this;
		}


		self operator--(int)
		{
			self tmp(*this);
			_it++;
			return tmp;
		}


		Ref operator*()
		{
			return *_it;
		}


		Ptr operator->()
		{
			return _it;
		}


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


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

	template<typename 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;
		typedef Reverse_iterator<iterator, T&, T*> reverse_iterator;
		typedef Reverse_iterator< const_iterator, const T&, const T*> reverse_const_iterator;


		//迭代器部分
		iterator begin()
		{
			return _head->_next;
		}

		iterator end()
		{
			return _head;
		}

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

		const_iterator end()const
		{
			return _head;
		}

		reverse_iterator rbegin()
		{
			return (--end());//_head->_prev
		}

		reverse_iterator rend()
		{
			return (end());//_head
		}

		reverse_const_iterator rbegin()const
		{
			return (--end());//_head->_prev
		}

		reverse_const_iterator rend()const
		{
			return (end());//_head
		}

		/// //
		/// 
	private:
		//不希望外界调用,设计成私有
		void empty_init()
		{
			_head = new Node;
			_head->_next = _head;
			_head->_prev = _head;
			_size = 0;
		}
	public:
		//构造、析构部分
		list()
		{
			empty_init();
		}

		list(size_t n, const T& value = T())
		{
			empty_init();
			while (n--)
			{
				push_back(value);
			}
		}

		//重载给内置类型使用,整形默认是int,不写这个会优先匹配list(Iterator first, Iterator last)
		list(int n, const T& value = T())
		{
			empty_init();
			while (n--)
			{
				push_back(value);
			}
		}

		//迭代器区间初始化
		template <class Iterator>
		list(Iterator first, Iterator last)
		{
			empty_init();
			while(first != last)
			{
				push_back(*first);
				first++;
			}
		}

		list(const list<T>& lt)
		{
			empty_init();
			for (auto e : lt)
			{
				push_back(e);
			}
		}

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


		//其它
		void swap(list<T> lt)
		{
			std::swap(_size, lt._size);
			std::swap(_head, lt._head);
		}

		//使用传之传参,直接拷贝一份交换操作的底层空间就好
		list<T>& operator=(list<T> lt)
		{
			swap(lt);
			return *this;
		}

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

	
		/// /
		/

		//访问头,尾数据
		T& front()
		{
			return _head->_next->_data;
		}

		const T& front()const
		{
			return _head->_next->_data;
		}

		T& back()
		{
			return _head->_prev->_data;
		}

		const T& back()const
		{
			return _head->_prev->_data;
		}



		/// //
		/// 

		//增加删除部分
		void push_back(const T& val)
		{
			insert(end(), val);
		}

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

		iterator insert(iterator pos, const T& val)
		{
			Node* newnode = new Node(val);
			Node* cur = pos._node;
			Node* prev = cur->_prev;

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

			++_size;
			return newnode;
		}

		void pop_back()
		{
			erase(--end());
		}

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

		iterator erase(iterator pos)
		{
			Node* cur = pos._node;
			Node* prev = cur->_prev;
			Node* next = cur->_next;

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

			--_size;
			return next;
		}

		//获取有效数据量
		size_t size()
		{
			return _size;
		}

	private:
		Node* _head;  //这里存储卫兵节点,因为底层是双向循环链表,可以找到头和尾
		size_t _size; //只需要在insert和erase里面加减就可以
	};

}

到了这里,关于C++:list使用以及模拟实现的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • C++初阶:适合新手的手撕list(模拟实现list)

    上次讲了常用的接口:今天就来进行模拟实现啦 list.h头文件:包含类的全部(函数的声明与定义) reverseIterator.h文件:进行反向迭代器的实现 test.cpp源文件:进行调用test函数,测试和完善功能 基本结构: ListNode 结构体: 定义了链表的节点结构,包含了三个成员变量:前驱指针

    2024年02月20日
    浏览(50)
  • C++初阶之一篇文章教会你list(模拟实现)

    成员类型表 这个表中列出了C++标准库中list容器的一些成员类型定义。这些类型定义是为了使list能够与C++标准库的其他组件协同工作,并提供一些通用的标准接口。每个成员类型的用处: value_type : 这个成员类型代表list容器中存储的数据类型,即模板参数T的类型。 allocator_

    2024年02月12日
    浏览(40)
  • 【C++】容器篇(二)——List的基本概述以及模拟实现

    前言: 在上期,我们学习了STL库中的第一个容器--vector ,今天我将给大家介绍的是 库中的另外一个容器--List。其实,有了之前学习 vector 的知识,对于List 的学习成本就很低了。 目录 (一)基本介绍 1、基本概念 2、list 与 forward_list 的比较 3、特点 (二)list的使用 1、list的

    2024年02月06日
    浏览(45)
  • 【C++进阶(五)】STL大法--list模拟实现以及list和vector的对比

    💓博主CSDN主页:杭电码农-NEO💓   ⏩专栏分类:C++从入门到精通⏪   🚚代码仓库:NEO的学习日记🚚   🌹关注我🫵带你学习C++   🔝🔝 本篇文章立足于上一篇文章: list深度剖析(上) 请先阅读完上一篇文章后再阅读这篇文章! 本章重点: 本章着重讲解list的模拟实现 list模拟实

    2024年02月09日
    浏览(53)
  • 【C++历练之路】list的重要接口||底层逻辑的三个封装以及模拟实现

    W...Y的主页 😊 代码仓库分享💕  🍔前言: 在C++的世界中,有一种数据结构,它不仅像一个神奇的瑰宝匣,还像一位能够在数据的海洋中航行的智慧舵手。这就是C++中的list,一个引人入胜的工具,它以一种优雅而强大的方式管理着数据的舞台。想象一下,你有一个能够轻松

    2024年02月04日
    浏览(42)
  • 【C++初阶】10. vector的使用及模拟实现

    vector的文档介绍 vector是表示可变大小数组的序列容器。 就像数组一样,vector也采用的连续存储空间来存储元素。也就是意味着可以采用下标对vector的元素进行访问,和数组一样高效。但是又不像数组,它的大小是可以动态改变的,而且它的大小会被容器自动处理。 必要时查

    2024年02月06日
    浏览(51)
  • 【C++】list的使用与模拟实现

    目录 一、list介绍 二、list的使用  1、list的构造 2、list capacity 3、list element access 4、list iterator 5、list modifiers 5.1、insert 6、list Operations 6.1、sort 7、list的迭代器失效 三、list模拟实现 1、push_back 2、iterator 3、const iterator 4、Ptr 5、insert 及其复用 6、erase 及其复用 7、clear 8、构造函

    2023年04月26日
    浏览(40)
  • 【C++】list的使用和模拟实现

    list的底层是一个双向带头循环链表,双向链表中每个元素存储在互不相关的独立节点中,在节点中通过指针指向其前一个元素和后一个元素,其遍历只能通过迭代器来实现,范围for的底层也是迭代器。 迭代器是所有容器都可以使用的迭代方式。 与list类似的还有forward_list,底

    2024年02月09日
    浏览(49)
  • C++:List的使用和模拟实现

                                                            创作不易,感谢三连!! list的文档介绍 1. list是可以 在常数范围内在任意位置进行插入和删除的序列式容器 ,并且该容器 可以前后双向迭代。 2. list的 底层是双向链表结构, 双向链表中每个元素存储在互不相

    2024年03月25日
    浏览(40)
  • [C++入门]---List的使用及模拟实现

    list 是可以在常数范围内在任意位置进行插入和删除的序列式容器,并且该容器可以前后双向迭代。 list 的底层是双向链表结构,双向链表中每个元素存储在互不相关的独立节点中,在节点中通过指针指向 其前一个元素和后一个元素。 list 与 forward_list 非常相似:最主要的不

    2024年02月07日
    浏览(35)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包