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

2024-08-26 06:52

本文主要是介绍移情别恋c++ ദ്ദി˶ー̀֊ー́ ) ——7.list(模拟实现),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

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到底是什么类型时统一调用打印函数

这篇关于移情别恋c++ ദ്ദി˶ー̀֊ー́ ) ——7.list(模拟实现)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1107855

相关文章

C#借助Spire.XLS for .NET实现在Excel中添加文档属性

《C#借助Spire.XLSfor.NET实现在Excel中添加文档属性》在日常的数据处理和项目管理中,Excel文档扮演着举足轻重的角色,本文将深入探讨如何在C#中借助强大的第三方库Spire.... 目录为什么需要程序化添加Excel文档属性使用Spire.XLS for .NET库实现文档属性管理Sp

C++ move 的作用详解及陷阱最佳实践

《C++move的作用详解及陷阱最佳实践》文章详细介绍了C++中的`std::move`函数的作用,包括为什么需要它、它的本质、典型使用场景、以及一些常见陷阱和最佳实践,感兴趣的朋友跟随小编一起看... 目录C++ move 的作用详解一、一句话总结二、为什么需要 move?C++98/03 的痛点⚡C++

Python+FFmpeg实现视频自动化处理的完整指南

《Python+FFmpeg实现视频自动化处理的完整指南》本文总结了一套在Python中使用subprocess.run调用FFmpeg进行视频自动化处理的解决方案,涵盖了跨平台硬件加速、中间素材处理... 目录一、 跨平台硬件加速:统一接口设计1. 核心映射逻辑2. python 实现代码二、 中间素材处

Java数组动态扩容的实现示例

《Java数组动态扩容的实现示例》本文主要介绍了Java数组动态扩容的实现示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录1 问题2 方法3 结语1 问题实现动态的给数组添加元素效果,实现对数组扩容,原始数组使用静态分配

Python实现快速扫描目标主机的开放端口和服务

《Python实现快速扫描目标主机的开放端口和服务》这篇文章主要为大家详细介绍了如何使用Python编写一个功能强大的端口扫描器脚本,实现快速扫描目标主机的开放端口和服务,感兴趣的小伙伴可以了解下... 目录功能介绍场景应用1. 网络安全审计2. 系统管理维护3. 网络故障排查4. 合规性检查报错处理1.

Python轻松实现Word到Markdown的转换

《Python轻松实现Word到Markdown的转换》在文档管理、内容发布等场景中,将Word转换为Markdown格式是常见需求,本文将介绍如何使用FreeSpire.DocforPython实现... 目录一、工具简介二、核心转换实现1. 基础单文件转换2. 批量转换Word文件三、工具特性分析优点局

Springboot3统一返回类设计全过程(从问题到实现)

《Springboot3统一返回类设计全过程(从问题到实现)》文章介绍了如何在SpringBoot3中设计一个统一返回类,以实现前后端接口返回格式的一致性,该类包含状态码、描述信息、业务数据和时间戳,... 目录Spring Boot 3 统一返回类设计:从问题到实现一、核心需求:统一返回类要解决什么问题?

详解C++ 存储二进制数据容器的几种方法

《详解C++存储二进制数据容器的几种方法》本文主要介绍了详解C++存储二进制数据容器,包括std::vector、std::array、std::string、std::bitset和std::ve... 目录1.std::vector<uint8_t>(最常用)特点:适用场景:示例:2.std::arra

C++构造函数中explicit详解

《C++构造函数中explicit详解》explicit关键字用于修饰单参数构造函数或可以看作单参数的构造函数,阻止编译器进行隐式类型转换或拷贝初始化,本文就来介绍explicit的使用,感兴趣的可以... 目录1. 什么是explicit2. 隐式转换的问题3.explicit的使用示例基本用法多参数构造

Java使用Spire.Doc for Java实现Word自动化插入图片

《Java使用Spire.DocforJava实现Word自动化插入图片》在日常工作中,Word文档是不可或缺的工具,而图片作为信息传达的重要载体,其在文档中的插入与布局显得尤为关键,下面我们就来... 目录1. Spire.Doc for Java库介绍与安装2. 使用特定的环绕方式插入图片3. 在指定位