C++从入门到起飞之——list模拟实现 全方位剖析!

news2024/9/25 13:26:01

​ ​

 

🌈个人主页:秋风起,再归来~
🔥系列专栏:C++从入门到起飞          
🔖克心守己,律己则安

目录

​ ​1、list的整体框架

2、list迭代器

>整体分析

>整体框架

 >成员函数

>运算符->重载 

>const迭代器

>迭代器类的一些问题的思考

3、成员函数

4、完结散花


1、list的整体框架


list就是我们所熟悉的数据结构链表,通过查看源码以及我们对链表的了解我们知道在类模版list中的成员变量一个是指向有效链表的哨兵卫头结点_head,另一个是记录有效元素个数的_size,而链表中的有效元素的存储地址空间不是连续的,所以并没有容量capacity的概念!
而我们又知道链表是由一个又一个的节点组成的,而我们要实现的是带头双向循环链表,所以每个节点里面都存储着数据val、前驱指针pre、后继指针next!
所以我们可以先大概写出list的整体框架!

//链表的节点
template<class T>
struct list_node
{
    list_node(const T& val = T())
        :data(val)
        ,pre(nullptr)
        ,next(nullptr)
    {}
    T data;//数据
    list_node<T>* pre;//前驱指针
    list_node<T>* next;//后继指针
};
/双向循环带头链表
template<class T>
class list
{
public:
    typedef list_node<T> node;//给节点取别名
    //成员函数
    //………………
    
private:
    node* _head=nullptr;//带哨兵卫的头结点
    size_t _size = 0;//记录节点个数
};

2、list迭代器


>整体分析


为了后续方便遍历容器打印结果检测我们程序的正确性,我们先来实现list的迭代器!
我们知道我们之前模拟的string和vector的迭代器都是原生的指针,那是因为他们底层结构的优越性允许他们这样实现。但是,list如果我们还是使用原生指针的话就完全做不到迭代器的效果,因为我们对节点指针*取到的是节点本身,而不是节点里面的数据,对节点++或--,因为链表不是连续的物理空间,所以我们并不能拿到前一个或后一个的迭代器反而让迭代器成了野指针!
那怎么办呢!我们可以对list的迭代器进行封装,而迭代器的底层原理还是使用节点的指针!只不过,我们要在迭代器里面对*、++、--、==、!= 等操作进行运算符重载!


>整体框架


那我们先实现迭代器的整体框架!
因为该迭代器只是给list使用,外面根本不知道,所以我们干脆用struct定义,将所有成员公开!

//链表的迭代器
template<class T>
struct list_iterator
{
    typedef  list_node<T> node;//给节点取别名
    typedef     list_iterator<T> self;//给自己取别名
    node* it;//链表中的迭代器其实就是一个节点的指针
             //只不过其访问和迭代操作要用一个类封装起来
    list_iterator(node* node)
        :it(node)
    {}
    //成员函数
    //………………

};

 >成员函数

//返回T&
T& operator*()
{
    return it->data;
}

//(前置++)++it
self& operator++()
{
    it = it->next;
    return *this;
}


//(前置--)--it
self& operator--()
{
    it = it->pre;
    return *this;
}

//(后置++) it++
self operator++(int) 
{
    self tmp(*this);
    it = it->next;
    return tmp;//一定不能用引用接收,不然会有野指针!
}

//(后置--) it--
self operator--(int) 
{
    self tmp(*this);
    it = it->pre;
    return tmp;
}

//比较迭代器里面的it指针相不相等
bool operator==(const self& comp)
{
    return it==comp.it;
}
bool operator!=(const self& comp)
{
    return it!=comp.it;
}


>运算符->重载 

如果节点中存储的是自定义类型Pos(坐标类)

struct Pos
{
    int _row;
    int _col;
    Pos(int row, int col)
        :_row(row)
        , _col(col)
    {}
};

​如果我们想要方便访问坐标,那我们可以重载->操作符

T* operator->()
{
    return &(it->data);
}
list<Pos> lt;
lt.push_back(Pos(1, 1)); 
lt.push_back(Pos(2, 2));
lt.push_back(Pos(3, 3));
lt.push_back(Pos(4, 4));

auto it = lt.begin();
while (it != lt.end())
{
    cout << it->_col << " " << it->_row << endl;
    ++it;
}
​

上面的->实际上的完整调用是下面的模式

auto it = lt.begin();
while (it != lt.end())
{
    cout << it.operator->()->_row << " " << it.operator->()->_col << endl;
    ++it;
}

it先调用重载的->,返回Pos类型的原生指针指针后再调用原始指针的->。而为了方便与可读性,编译器允许我们省略一个->调用! 


>const迭代器

我们在上面只实现了一个普通的迭代器,如果我们要再实现一个const的常量迭代器,我们是不是就要再自己实现一个迭代器类呢?雀氏行的通!

//链表的迭代器
template<class T>
struct list_const_iterator
{
    typedef  list_node<T> node;//给节点取别名
    typedef     list_const_iterator<T> self;//给自己取别名
    node* it;//链表中的迭代器其实就是一个节点的指针
             //只不过其访问和迭代操作要用一个类封装起来
    list_const_iterator(node* node)
        :it(node)
    {}
    //成员函数
    //………………

};

不过,在内部的成员函数实现中,只有一些函数的返回值是不同的,而函数体的是一样的!想这样俩个相似度高度集中的类,我们就可以使用类模版的方法将俩个类进行合并!
而我们具体的做法则是增加模版参数!

合并后的完整代码:

//链表的迭代器
template<class T, class ref,class ptr>
struct list_iterator
{
    typedef  list_node<T> node;//给节点取别名
    typedef     list_iterator<T, ref, ptr> self;//给自己取别名
    node* it;//链表中的迭代器其实就是一个节点的指针
             //只不过其访问和迭代操作要用一个类封装起来
    list_iterator(node* node)
        :it(node)
    {}

    //返回T*
    ptr operator->()
    {
        return &(it->data);
    }

    //返回T&
    ref operator*()
    {
        return it->data;
    }

    //(前置++)++it
    self& operator++()
    {
        it = it->next;
        return *this;
    }
    /*self operator++()
    {
        it = it->next;
        return it;
    }*/

    //(前置--)--it
    self& operator--()
    {
        it = it->pre;
        return *this;
    }
    /*self operator--()
    {
        it = it->pre;
        return it;
    }*/

    //(后置++) it++
    self operator++(int) 
    {
        self tmp(*this);
        it = it->next;
        return tmp;//一定不能用引用接收,不然会有野指针!
    }

    //(后置--) it--
    self operator--(int) 
    {
        self tmp(*this);
        it = it->pre;
        return tmp;
    }

    //比较迭代器里面的it指针相不相等
    bool operator==(const self& comp)
    {
        return it==comp.it;
    }
    bool operator!=(const self& comp)
    {
        return it!=comp.it;
    }

};

>迭代器类的一些问题的思考

(1) 类中是否需要写析构函数?
这个迭代器类不要写析构函数,因为这里的节点不是迭代器的,是链表的,不用把它释放。我们使用begin,end返回节点给迭代器,是借助迭代器修改,访问数据,所以我们不需要释放。
(2) 类中是否需要写拷贝构造进行深拷贝和写赋值拷贝?
这里也不需要写拷贝构造进行深拷贝,因为这里要的就是浅拷贝。begin返回了第一个节点的迭代器给it,这里就是用默认生成的拷贝构造,浅拷贝给it,那这两个迭代器就指向同一个节点,所以这里用默认的拷贝构造和赋值拷贝就可以了。

3、成员函数

对于list的常见成员函数的实现我们已近轻车熟路了!这里就直接给源码了!

//链表的节点
template<class T>
struct list_node
{
	list_node(const T& val = T())
		:data(val)
		,pre(nullptr)
		,next(nullptr)
	{}
	T data;//数据
	list_node<T>* pre;//前驱指针
	list_node<T>* next;//后继指针
};

//链表的迭代器
template<class T, class ref,class ptr>
struct list_iterator
{
	typedef  list_node<T> node;//给节点取别名
	typedef	 list_iterator<T, ref, ptr> self;//给自己取别名
	node* it;//链表中的迭代器其实就是一个节点的指针
			 //只不过其访问和迭代操作要用一个类封装起来
	list_iterator(node* node)
		:it(node)
	{}

	//返回T*
	ptr operator->()
	{
		return &(it->data);
	}

	//返回T&
	ref operator*()
	{
		return it->data;
	}

	//(前置++)++it
	self& operator++()
	{
		it = it->next;
		return *this;
	}
	/*self operator++()
	{
		it = it->next;
		return it;
	}*/

	//(前置--)--it
	self& operator--()
	{
		it = it->pre;
		return *this;
	}
	/*self operator--()
	{
		it = it->pre;
		return it;
	}*/

	//(后置++) it++
	self operator++(int) 
	{
		self tmp(*this);
		it = it->next;
		return tmp;//一定不能用引用接收,不然会有野指针!
	}

	//(后置--) it--
	self operator--(int) 
	{
		self tmp(*this);
		it = it->pre;
		return tmp;
	}

	//比较迭代器里面的it指针相不相等
	bool operator==(const self& comp)
	{
		return it==comp.it;
	}
	bool operator!=(const self& comp)
	{
		return it!=comp.it;
	}

};

//双向循环带头链表
template<class T>
class list
{
public:
	typedef list_node<T> node;//给节点取别名
	typedef list_iterator<T, T&, T*> iterator;//普通迭代器
	typedef list_iterator<T, const T&, const T*> const_iterator;//常量迭代器

	//size
	size_t size() const
	{
		return _size;
	}

	//begin
	iterator begin()
	{
		return _head->next;
	}

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

	//cbegin
	const_iterator begin() const
	{
		return _head->next;
	}

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

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

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

	//empty
	bool empty()
	{
		return _size == 0;
	}

	//空初始化
	void emptyInit()
	{
		_head = new node;
		_head->next = _head->pre = _head;
		_size = 0;
	}
	
	//无参的默认构造
	list()
	{
		emptyInit();
	}

	//拷贝构造
	list(const list<T>& lt)
	{
		emptyInit();//先空参构造_head
		for (auto& e : lt)
		{
			push_back(e);
		}
		/*auto it = lt.begin();
		while (it != lt.end())
		{
			push_back(*it);
			++it;
		}*/
	}

	//赋值运算符重载
	list<T>& operator=(list<T> tmp)
	{
		swap(tmp);
		return *this;
	}

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

	//insert
	iterator insert(iterator pos, const T& val)
	{
		node* newNode = new node(val);
		node* cur = pos.it;
		node* pre = cur->pre;

		newNode->next = cur;
		newNode->pre = pre;
		pre->next = newNode;
		cur->pre = newNode;

		_size++;
		return newNode;
	}
	
	//push_front(头插)
	void push_front(const T& val)
	{
		insert(begin(), val);
	}

	//push_back(尾插)
	void push_back(const T& val)//传引用减少拷贝构造
	{
		/*node* newNode = new node(val);
		node* tail = _head->pre;

		tail->next = newNode;
		newNode->next = _head;
		newNode->pre = tail;
		_head->pre = newNode;

		++_size;*/

		insert(end(), val);
	}

	//erase
	iterator erase(iterator pos)
	{
		assert(pos != end());
		node* next = pos.it->next;
		node* pre = pos.it->pre;

		pre->next = next;
		next->pre = pre;
		delete pos.it;
		pos.it = nullptr;
		_size--;
		return next;
	}

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

	//pop_back(尾删)
	void pop_back()
	{
		erase(--end());
	}
	
private:
	node* _head=nullptr;//带哨兵卫的头结点
	size_t _size = 0;//记录节点个数
};

4、完结散花

好了,这期的分享到这里就结束了~

如果这篇博客对你有帮助的话,可以用你们的小手指点一个免费的赞并收藏起来哟~

如果期待博主下期内容的话,可以点点关注,避免找不到我了呢~

我们下期不见不散~~

​​​​​​

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

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

相关文章

操作系统:实验六文件操作实验

一、实验目的 1、了解文件系统功能及实现原理。 2、掌握LINUX下文件操作的有关系统调用。 3、熟悉main函数带参数运行的有关操作过程。 4、通过模拟程序实现简单的一级文件系统或二级文件系统。 二、实验内容 1、编程显示文件自身。&#xff08;1分&#xff09; #includ…

分享两个方法分析python打包exe

在Python开发中&#xff0c;常常需要将Python脚本打包成独立的可执行文件&#xff0c;以便在没有Python环境的电脑上运行。你是否曾为此感到困惑&#xff0c;不知道该选择哪种工具来实现这一目标&#xff1f;其实&#xff0c;打包Python脚本并不难&#xff0c;关键在于选择合适…

Mybatis框架——缓存(一级缓存,二级缓存)

本章将简单介绍Mybatis框架中的缓存&#xff0c;欢迎大家点赞➕收藏&#xff0c;蟹蟹&#xff01;&#xff01;&#xff01;&#x1f495; &#x1f308;个人主页&#xff1a;404_NOT_FOUND &#x1f308;MyBatis环境搭建步骤&#xff08;超全解析&#xff01;&#xff01;&am…

AI写论文真的可靠吗?免费推荐6款AI论文写作助手

在当今的学术研究和写作领域&#xff0c;AI论文写作助手已经成为不可或缺的工具。这些工具不仅能够提高写作效率&#xff0c;还能帮助研究者生成高质量的论文。以下是六款免费推荐的AI论文写作助手&#xff0c;包括千笔-aipasspaper&#xff0c;它们各自具有独特的功能和优势。…

【hot100篇-python刷题记录】【最小路径和】

R6-多维动态规划篇 好经典的dp题&#xff0c;纯粹的题。 多维动态规划无论是二维还是三维&#xff0c;无非是创建dp表&#xff0c;dp[][][][][][]即可 动态规划式子 dp[i][j]当前值min(dp[i][j-1],dp[i-1][j]) 边界问题处理&#xff1a;是否存在即可。哦对了好像不用这样&a…

探索异步之美:aiohttp库的魔力与奥秘

文章目录 探索异步之美&#xff1a;aiohttp库的魔力与奥秘背景&#xff1a;为何选择aiohttp&#xff1f;什么是aiohttp&#xff1f;如何安装aiohttp&#xff1f;简单函数使用方法场景应用常见Bug及解决方案总结 探索异步之美&#xff1a;aiohttp库的魔力与奥秘 背景&#xff1…

Linux教程七:文件目录类命令ls、cd(图文详解)

默认登录出现一个[用户localhost ~] 代表时登陆用户的家目录 1、 Linux ls命令 基本用法 ls&#xff1a;列出当前目录下的文件和目录&#xff08;不包括以.开头的隐藏文件&#xff09;。ls 目录名&#xff1a;列出指定目录下的文件和目录。 常用选项 -l&#xff1a;以长格式列出…

【Python 报错已解决】`TypeError: ‘method‘ object is not subscriptable`

&#x1f3ac; 鸽芷咕&#xff1a;个人主页 &#x1f525; 个人专栏: 《C干货基地》《粉丝福利》 ⛺️生活的理想&#xff0c;就是为了理想的生活! 引言 在编程的世界里&#xff0c;我们经常会遇到各种报错&#xff0c;它们像隐藏在代码中的小怪兽&#xff0c;时不时跳出来给…

Java项目:基于SpringBoot+mysql在线拍卖系统(含源码+数据库+答辩PPT+毕业论文)

一、项目简介 本项目是一套基于SSM框架mysql在线拍卖系统 包含&#xff1a;项目源码、数据库脚本等&#xff0c;该项目附带全部源码可作为毕设使用。 项目都经过严格调试&#xff0c;eclipse或者idea 确保可以运行&#xff01; 该系统功能完善、界面美观、操作简单、功能齐全、…

常见的性能测试方法!

前言 性能测试划分有很多种&#xff0c;测试方法也有很多种&#xff0c;更确切的说是由于测试方法的不同决定了测试划分的情况&#xff0c;但在测试过程中性能测试的划分没有绝对的界限&#xff0c;常用的有压力测试、负载测试和并发用户测试等。 性能测试的方法主要包括以下…

划分字母区间

划分字母区间 思路&#xff1a; 我觉得这道题最关键的一个思路就是&#xff0c;对于没一个字母&#xff0c;你一定要找到他的最后一次出现的位置&#xff0c;每一个片段必须要比这个大&#xff0c;然后该字符到这个end中其他的字符&#xff0c;也都要找到最后一次出现的位置&…

net6 core 接入nacos 实现服务注册入门使用,心跳检测和负载均衡

net6 core 接入nacos 实现服务注册入门使用&#xff0c;心跳检测和负载均衡 配置中心比较Apollo与Nacos_appollo 和 nacos-CSDN博客 一&#xff1a;安装nacos Release 2.2.3 (May 25th, 2023) alibaba/nacos GitHub 二、配置Nacos 注*Nacos 是使用的mysql 数据库&#x…

谷粒商城实战笔记-282~283-商城业务-订单服务-提交订单的问题

文章目录 一&#xff0c;282-商城业务-订单服务-提交订单的问题调试过程中出现的问题services面板介绍什么是 Services Panel&#xff1f;主要作用解决的痛点使用方法 二&#xff0c;283-商城业务-分布式事务-本地事务在分布式下的问题分布式事务问题解决方案分布式事务处理流程…

Hubspot AI 工具| 使用 6 款 HubSpot AI 工具,提升初创团队海外营销与销售效率

看看 HubSpot AI 工具如何完美助力中国出海与外贸企业加速落地全球 GTM 策略吧~ 在日益竞争激烈的全球市场中&#xff0c;初创企业想要获得成功&#xff0c;必须有效将产品推向市场&#xff0c;并建立客户基础&#xff0c;与竞争对手一较高下。 这需要精心策划的进入市场&…

常用于单北斗多频定位导航模块资料:ATGM332D-F8N

单北斗多频定位导航模块的高精度定位优点描述&#xff1a; 1、多频信号融合&#xff1a;单北斗多频定位导航模块能够接收和处理来自北斗卫星系统的多个频段信号。通过多频信号的融合处理&#xff0c;可以显著提高定位的精度。因为不同频段的信号在传播过程中受到的影响不同&…

力扣134-加油站(java题解)

题目链接&#xff1a;134. 加油站 - 力扣&#xff08;LeetCode&#xff09; 前情提要&#xff1a; 因为本人最近都来刷贪心类的题目所以该题就默认用贪心方法来做。 贪心方法&#xff1a;局部最优推出全局最优。 如果一个题你觉得可以用局部最优推出全局最优&#xff0c;并…

IO进程day06(进程间通信、信号、共享内存)

目录 【1】进程间通信 IPC 1》 进程间通信方式 2》 无名管道 1> 特点 2> 函数接口 3> 注意事项 练习&#xff1a;父子进程实现通信&#xff0c;父进程循环从终端输入数据&#xff0c;子进程循环打印数据&#xff0c;当输入quit结束。 3》有名管道 1> 特点 …

24数学建模国赛准备!!!!(10——马氏链模型)

详细获取资料方式在文章末尾&#xff01;&#xff01;&#xff01;&#xff01; 点击链接加入群聊获取资料以及国赛助力https://qm.qq.com/q/NGl6WD0Bky &#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&…

多场景建模: STAR(Star Topology Adaptive Recommender)

之前&#xff0c;分享了一篇关于多任务学习的文章&#xff1a;多任务学习MTL模型&#xff1a;MMoE、PLE&#xff0c;同样的还有关于多任务学习中的多目标loss优化策略。 这篇文章则开始一个与多任务学习有着紧密联系的系列&#xff1a;多场景建模学习。 前言 首先&#xff0…

[Raspberry Pi]如何利用docker執行motioneye,並利用Line Notify取得即時通知和照片?

[Motioneye]How to setup motion detection and send message/image for Line Notify 無意間&#xff0c;翻了一本關於樹莓派的書籍&#xff0c;除了樹莓派的簡介和應用外&#xff0c;也包含初階和高階的Linux運作邏輯&#xff0c;書籍結構相當完整&#xff0c;也因此需要花時間…