移情别恋c++ ദ്ദി˶ー̀֊ー́ ) ——7.list(模拟实现)

news2025/1/17 5:55:43

1.前言

1.1list与vector的不同

区别:list的迭代器底层和其他两个迭代器底层有很大区别,因为list的链式结构决定了与它们两个的不一样

相同:迭代器用法大致一样,其他成员函数的使用也大致一样。

vector与list都是STL中非常重要的序列式容器,由于两个容器的底层结构不同,导致其特性以及 应用场景不同,其主要不同如下

1.2 迭代器的分类 

 例子:

可以得知list类型无法使用std中的sort函数,因为list的迭代器是双向的,而sort函数的迭代器参数是随机的 ,同理可以得出,string,vector,deque都可以使用std中的sort

 1.3 list的本质

带头双向循环链表!!!!!!!!!!!!!!!!!!!

2.list节点 

template<class T>//记得每一个类前要写模板
struct list_node       //struct和class一样均为类,不同的是struct中的所有成员均为公有(public)类型,可直接访问
{
	T data;
	list_node<T>* next;
	list_node<T>* prev;


	list_node(const T& x=T())  //const T& x=T(),在没有给值的情况下,系统会通过T()自动生成一个类型匹配的值赋给x
		:data(x)
		,next(nullptr)
		,prev(nullptr)
	{}


};

用类来封装一个一个结点,里面有两个指针,一个是指向下一个位置的指针,一个是指向前一个位置,还有一个用来存放数据的变量

3.list类框架 

template<class T>
class List
{
 public:
	typedef List_node<T> node;//取别名
    void empty_list()
	{
		head = new Node;
		head->_next = head;
		head->_prev = head;
	}
	List()//构造函数
	{
	    empty_list();
	}
 private:
     node*head;//头节点
     size_t _size;
}

list类里面包含的是结点的指针,也就是哨兵位头节点的指针

4.list迭代器

4.1 list迭代器框架

这里的迭代器是用封装加运算符重载来实现的,由于迭代器也会有很多类型,所以我们使用模板的形式

//T,T&,T*
//T,const T&,const T*        //设定了两种迭代器
template<class T,class Ref,class Ptr>//记得每一个类前要写模板,可设置多个模板参数
struct list_iterator
{
	typedef list_node<T> node;
	typedef list_iterator<T,Ref,Ptr> self;
	node* node1;


	list_iterator(node* node2)
		:node1(node2)
	{}
}

4.2 list常用迭代器 

//T,T&,T*
//T,const T&,const T*        //设定了两种迭代器

template<class T,class Ref,class Ptr>//记得每一个类前要写模板,可设置多个模板参数
struct list_iterator
{
	typedef list_node<T> node;
	typedef list_iterator<T,Ref,Ptr> self;
	node* node1;


	list_iterator(node* node2)
		:node1(node2)
	{}

	self& operator++()
	{
		node1 = node1->next;
		return *this;           //模拟++
	}

	self& operator--()
	{
		node1 = node1->prev;
		return *this;           //模拟--
	}


	Ref operator*()
	{
		return node1->data;    //模拟指针解引用
	}


	Ptr operator->()
	{
		return &node1->data;
	}

	bool operator!=(const self& S)
	{
		return node1 != S.node1;
	}

};

迭代器的每一个操作都采用了运算符重载,其实这样看来还是指针在操作,只不过他不是直接的进行操作,而是换了一种方式

5.list函数详解 

5.1插入和删除

void push_back(const T& x)  //尾插
{
	insert(end(), x);
}

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

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

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

iterator insert(iterator pos, const T& x)//在pos位置插入x
{
	node* it = pos.node1;
	node* newnode = new node(x);
	node* prev = it->prev;
	prev->next = newnode;
	newnode->prev = prev;
	newnode->next = it;
	it->prev = newnode;

	++_size;

	return iterator(newnode);                    //返回新插入的元素的位置
}

iterator erase(iterator pos)
{
	node* it = pos.node1;
	node* prev = it->prev;
	node* next = it->next;

	delete it;
	prev->next = next;
	next->prev = prev;
	--_size;

	return iterator(next);       //返回删除后的当前位置
}

5.2  拷贝构造和赋值运算符重载

list(const list<T>& it)    //i1(i2)
{
	int i = 0;
	empty_list();  //初始化列表
	for (auto e : it)  //相当于自动调用迭代器,并解引用,最后迭代器再++
	{
		i = 1;
		push_back(e);
	}

}


void swap(list<T>& it)
{
	std::swap(head, it.head);
	std::swap(_size, it._size);
}

list operator=(list<T>it)// i1=i2      传参直接调用拷贝构造
{
	swap(it);
	return *this;
}


size_t size()
{
	return _size;
}

5.3 list析构函数 

~list()
{
	clear();
	delete head;
	head = nullptr;

}

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

}

6.打印函数 

//打印
template<typename Container>
void print_container(const Container& con)
{
	typename Container::const_iterator it = con.begin();
	while (it != con.end())
	{
		cout << *it << " ";
		++it;
	}
	cout << endl;
}

这里用的是typename,原因在于编译器在编译的时候,会去Container里面去找const_iterator这个类型,但是Container是一个模板,并没有实例化,所以编译器也不知道怎么定义,所以就在编译的时候就过不了,但是我们在前面加一个typename就会告诉编译器,先过,后面再来实例化,这样就可以解决问题了

适用于不知道list<T>中的T到底是什么类型时统一调用打印函数

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

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

相关文章

关于安装hbase的问题(操作系统-windows)

&#x1f3c6;本文收录于《CSDN问答解惑-专业版》专栏&#xff0c;主要记录项目实战过程中的Bug之前因后果及提供真实有效的解决方案&#xff0c;希望能够助你一臂之力&#xff0c;帮你早日登顶实现财富自由&#x1f680;&#xff1b;同时&#xff0c;欢迎大家关注&&收…

快速学习“堆“排序(C语言数据结构)

前言&#xff1a; 堆的实现其实并不难&#xff0c;难的是要用堆实现排序&#xff0c;也就是堆的运用。 下面需要探究一下堆的排序是怎样的。 如何利用堆进行升序或者降序的排序。 "堆排序"&#xff1a; 原理&#xff1a; 例如&#xff1a;此时要将数组里的数组int a…

干货实用帖 | PARASOFT与JENKINS 插件集成

&#x1f4d6; 介绍&#xff1a; 本篇介绍如何使用Jenkins上的插件Parasoft Findings&#xff0c;应用到C/Ctest项目中。 ✅ 准备工作&#xff1a; Jenkins项目C/Ctest 10.4以上版本及有效的许可证 视频教学&#xff1a; Parasoft与Jenkins插件集成 安装插件&#xff1a; 首先…

Vue3 获取农历(阴历)日期,并封装日历展示组件

前言&#xff1a;哈喽&#xff0c;大家好&#xff0c;我是码喽的自我修养&#xff01;今天给大家分享vue3项目中使用 chinese-lunar-calendar 插件获取农历(阴历)日期&#xff0c;并封装了日历展示组件&#xff01;提供了具体的代码帮助大家深入理解&#xff0c;彻底掌握&#…

【舞动生命,营养护航】亨廷顿舞蹈症患者的维生素补给站

Hey小伙伴们~&#x1f44b; 在这个充满色彩的世界里&#xff0c;每个人都在以自己的方式绽放光彩。但你知道吗&#xff1f;有一群特别的朋友&#xff0c;他们面对着亨廷顿舞蹈症的挑战&#xff0c;却依然以不屈不挠的精神舞动着生命的旋律。&#x1f483;✨ 今天&#xff0c;就…

游戏如何对抗 IL2cppDumper逆向分析

众所周知&#xff0c;Unity引擎中有两种脚本编译器&#xff0c;分别是 Mono 和 IL2CPP 。相较于Mono&#xff0c;IL2CPP 具备执行效率高、跨平台支持等优势&#xff0c;已被大多数游戏采用。 IL2CPP 模式下&#xff0c;可以将游戏 C# 代码转换为 C 代码&#xff0c;然后编译为…

STM32学习记录-06-ADC模数转换器

1 ADC简介 ADC(Analog-Digital Converter)模拟-数字转换器 ADC可以将引脚上连续变化的模拟电压转换为内存中存储的数字变量,建立模拟电路到数字电路的桥梁 12位逐次逼近型ADC,1us转换时间 输入电压范围:0~3.3V,转换结果范围:0~4095 18个输入通道,可测量16个外部和…

FPGA在医疗方面的应用

可编程逻辑支持以灵活、低风险的方式成功实施系统设计&#xff0c;同时提供了最佳的成本效率和增值的差异化功能&#xff0c;延长了医疗保健应用的生命周期&#xff0c;包括诊断成像、电子医疗、治疗和生命科学与医院设备。 在医疗方面的应用非常广泛&#xff0c;以下是几个主…

Langchain Memory组件深度剖析:从对话基础到高级链式应用

文章目录 前言一、Langchain memory 记忆1.Memory 组件基本介绍2.Memory 组件的类型1.ChatMessageHistory2.ConversationBufferMemory3.ConversationBufferWindowMemory4.ConversationEntityMemory5.ConversationKGMemory6.ConversationSummaryMemory 二、长时记忆1.简单介绍2.…

Error: Can not import paddle core while this file exists

背景 因为工作需要&#xff0c;原来的项目部署的电脑被征用&#xff0c;重新换了一个新电脑&#xff0c;重装了系统&#xff0c;今天在给一个使用ocr的项目进行环境配置的时候发现&#xff0c;无论安装哪个版本的paddlepaddle&#xff0c;总是可以安装成功&#xff0c;但是导入…

Android CCodec Codec2 (四)C2Param - Ⅱ

这一篇内容我们来解答复杂参数定义过程中提出的疑问&#xff0c;本文有大量的模板和宏展开&#xff0c;请耐心阅读。 1、不含灵活数组的复杂结构体定义 DEFINE_AND_DESCRIBE_C2STRUCT和C2FIELD是不能分开的&#xff0c;使用时必须要按顺序依次调用这两个宏定义。宏定义比较复杂…

【机器学习-监督学习】神经网络与多层感知机

【作者主页】Francek Chen 【专栏介绍】 ⌈ ⌈ ⌈Python机器学习 ⌋ ⌋ ⌋ 机器学习是一门人工智能的分支学科&#xff0c;通过算法和模型让计算机从数据中学习&#xff0c;进行模型训练和优化&#xff0c;做出预测、分类和决策支持。Python成为机器学习的首选语言&#xff0c;…

java-Mybaits框架01

1.框架概念 在基础语言之上&#xff0c;对各种基础功能进行封装&#xff0c;方便开发者&#xff0c;提高开发效率&#xff1b; java后端框架 mybaits&#xff1a;对jdbc进行封装 Spring&#xff1a;对整个java后端架构进行管理。 SpringWeb&#xff1a;对web&#xff08;S…

vxe-grid 利用dayjs提供的方法来格式化, 计算二个日期之间的年数/年龄

1、安装dayjs pnpm add dayjs yarn add dayjs npm install dayjs 2、导入 import dayjs from dayjs; 3、vxe-grid列&#xff1a; export const UserColumns: VxeGridPropTypes.Columns [ ... {title: 年龄,width: 70,field: old,showOverflow: tooltip,align: center,sortabl…

android studio 设置gradle jdk

1. 左上角点击file 2. 按照如下点击&#xff1a; 3. 即可修改gradle jdk

EasyExcel文件导出简洁版

1. EasyExcel简介 EasyExcel是一个基于Java的简单、快速、lightweight的Excel处理库。它的主要特点包括: 轻量级设计: EasyExcel的jar包大小仅约1MB,相比较其他Excel处理库如Apache POI来说更加轻量。 采用内存友好的流式读写模式,无需一次性加载整个Excel文件到内存,大大减少…

Linux的CPU调度优化详解

一、引言 随着计算机硬件技术的不断发展和进步&#xff0c;现代服务器和工作站通常都配备了多核CPU&#xff0c;为了充分发挥多核处理器的性能优势&#xff0c;Linux系统提供了多种CPU调度器以及相关的参数设置&#xff0c;以便进行CPU调度优化&#xff0c;提高系统的整体性能…

AI 音频/文本对话机器人:Whisper+Edge TTS+OpenAI API构建语音与文本交互系统(简易版)

文章目录 前言思路&#xff1a;环境配置代码1. 加载Whisper模型2. 使用Whisper语音转文本3. 使用OpenAI API生成文本进行智能问答4. 实现文本转语音功能5. 合并音频文件6. 构建Gradio界面注意 总结 前言 在本篇博客中&#xff0c;我将分享如何利用Whisper模型进行语音转文本&a…

GFS系统架构

GFS系统架构 针对上述观察&#xff0c;我们发现它们与早期文件系统的设计假设存在显著差异。为此&#xff0c;我们采取了以下解决方案&#xff1a; 组件故障&#xff1a;我们接受故障为常态&#xff0c;系统设计以自我监控和快速恢复为原则&#xff0c;适应低成本硬件环境下的…

基础算法--递推算法[信奥一本通]

本节所讲题源自【信奥一本通】C版&#xff1a;基础算法-第三章-递推算法 相信大家应该都接触过数列的概念。哎哟&#xff0c;一直在跟数组打交道&#xff0c;说数列感觉好陌生&#xff0c;哈哈。数列中的迭代法大家都还记得吗&#xff1a;通过反复应用特定规则&#xff0c;推导…