移情别恋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

相关文章

python使用watchdog实现文件资源监控

《python使用watchdog实现文件资源监控》watchdog支持跨平台文件资源监控,可以检测指定文件夹下文件及文件夹变动,下面我们来看看Python如何使用watchdog实现文件资源监控吧... python文件监控库watchdogs简介随着Python在各种应用领域中的广泛使用,其生态环境也

el-select下拉选择缓存的实现

《el-select下拉选择缓存的实现》本文主要介绍了在使用el-select实现下拉选择缓存时遇到的问题及解决方案,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的... 目录项目场景:问题描述解决方案:项目场景:从左侧列表中选取字段填入右侧下拉多选框,用户可以对右侧

深入理解C++ 空类大小

《深入理解C++空类大小》本文主要介绍了C++空类大小,规定空类大小为1字节,主要是为了保证对象的唯一性和可区分性,满足数组元素地址连续的要求,下面就来了解一下... 目录1. 保证对象的唯一性和可区分性2. 满足数组元素地址连续的要求3. 与C++的对象模型和内存管理机制相适配查看类对象内存在C++中,规

Python pyinstaller实现图形化打包工具

《Pythonpyinstaller实现图形化打包工具》:本文主要介绍一个使用PythonPYQT5制作的关于pyinstaller打包工具,代替传统的cmd黑窗口模式打包页面,实现更快捷方便的... 目录1.简介2.运行效果3.相关源码1.简介一个使用python PYQT5制作的关于pyinstall

使用Python实现大文件切片上传及断点续传的方法

《使用Python实现大文件切片上传及断点续传的方法》本文介绍了使用Python实现大文件切片上传及断点续传的方法,包括功能模块划分(获取上传文件接口状态、临时文件夹状态信息、切片上传、切片合并)、整... 目录概要整体架构流程技术细节获取上传文件状态接口获取临时文件夹状态信息接口切片上传功能文件合并功能小

python实现自动登录12306自动抢票功能

《python实现自动登录12306自动抢票功能》随着互联网技术的发展,越来越多的人选择通过网络平台购票,特别是在中国,12306作为官方火车票预订平台,承担了巨大的访问量,对于热门线路或者节假日出行... 目录一、遇到的问题?二、改进三、进阶–展望总结一、遇到的问题?1.url-正确的表头:就是首先ur

C#实现文件读写到SQLite数据库

《C#实现文件读写到SQLite数据库》这篇文章主要为大家详细介绍了使用C#将文件读写到SQLite数据库的几种方法,文中的示例代码讲解详细,感兴趣的小伙伴可以参考一下... 目录1. 使用 BLOB 存储文件2. 存储文件路径3. 分块存储文件《文件读写到SQLite数据库China编程的方法》博客中,介绍了文

Redis主从复制实现原理分析

《Redis主从复制实现原理分析》Redis主从复制通过Sync和CommandPropagate阶段实现数据同步,2.8版本后引入Psync指令,根据复制偏移量进行全量或部分同步,优化了数据传输效率... 目录Redis主DodMIK从复制实现原理实现原理Psync: 2.8版本后总结Redis主从复制实

JAVA利用顺序表实现“杨辉三角”的思路及代码示例

《JAVA利用顺序表实现“杨辉三角”的思路及代码示例》杨辉三角形是中国古代数学的杰出研究成果之一,是我国北宋数学家贾宪于1050年首先发现并使用的,:本文主要介绍JAVA利用顺序表实现杨辉三角的思... 目录一:“杨辉三角”题目链接二:题解代码:三:题解思路:总结一:“杨辉三角”题目链接题目链接:点击这里

基于Python实现PDF动画翻页效果的阅读器

《基于Python实现PDF动画翻页效果的阅读器》在这篇博客中,我们将深入分析一个基于wxPython实现的PDF阅读器程序,该程序支持加载PDF文件并显示页面内容,同时支持页面切换动画效果,文中有详... 目录全部代码代码结构初始化 UI 界面加载 PDF 文件显示 PDF 页面页面切换动画运行效果总结主