c++中list的简单实现

文章目录

  • list
    • 介绍
    • 节点类(listNode)
    • __list__iterator(迭代器类)
      • operator->
    • list的成员函数
      • empty_init() 初始化节点
      • list(list<T>& lt) 拷贝构造
      • clear() 清除链表
      • ~list() 析构
      • insert() 插入
      • erase() 删除
      • push_back() 尾插
      • push_front() 头插
      • pop_back() 尾删
      • pop_front() 头删
      • begin() 头节点
      • end() 尾节点
  • 总结

list

介绍

list:
是数据结构中的链表,存储方式是在内存中每一个节点取一段空间用特定的方式链接起来,这样子就不会有浪费的空间
我们用的是带头循环双向链表

节点类(listNode)

因为一个节点中要包含其他信息所以单独弄成一个类

template<class T>
//链表节点类
struct listNode
{listNode<T>* _next;//指向下一个节点listNode<T>* _prev;//指向上一个节点T date;//内容listNode(const T& x = T()):_next(nullptr),_prev(nullptr),date(x){}
};

__list__iterator(迭代器类)

为什么要有迭代器类呢?
因为我们要封装一下这个迭代器,让迭代器该有的操作在list也可以用出来。
如果在list内部弄迭代器会很不好弄。

//正向迭代器类
//Ref 来区别const和普通
template<class T,class Ref,class Ptr>
struct __list__iterator
{typedef listNode<T> Node;//减少代码typedef __list__iterator<T,Ref> self;//来控制他的类别Node* _node;//节点__list__iterator(Node* node):_node(node){}//++itself& operator++(){_node = _node->_next;return *this;}//it++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->date;}bool operator!=(const self& s){return _node != s._node;}bool operator==(const self& s){return _node == s._node;}Ptr operator->(){return &_node->date;}
};

operator->

因为这个比较特殊很难看懂所以我们单独解释
他的本质是->->,代码解释更好看懂
因为之前的设计者觉得不好看,所以做的特殊处理

struct MyStruct
{int _a1;int _a2;MyStruct(int _a1 = 1, int _a2 = 1):_a1(_a1),_a2(_a2){}
};
void test2()
{list<MyStruct> s;s.push_back(MyStruct());s.push_back(MyStruct());s.push_back(MyStruct());//插入的是一个类list<MyStruct>::iterator lt = s.begin();while (lt != s.end()){cout << lt->_a1<<":"<< lt->_a2 << " ";//读这个类里面的内容//lt->_a1 的本质是 lt.operator->()->_a1; 特殊处理++lt;}
}

list的成员函数

因为我们偷点懒所以把一些东西typedef一下

	typedef listNode<T> Node;typedef __list__iterator<T,T&,T*> iterator;typedef __list__iterator<T,const T&,const T*> const_iterator;typedef Reverselterator<T, T&, T*> reverse_iterator;

empty_init() 初始化节点

因为我们要多次用到所以单独写一个出来方便
同时可以用list()套一下他

void empty_init()//初始化节点
{_head = new Node;_head->_next = _head;_head->_prev = _head;
}

list(list& lt) 拷贝构造

简单写法
引用的作用是深拷贝,不要弄成浅拷贝了

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

clear() 清除链表

直接把链表清空

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

~list() 析构

清除之后直接把哨兵位删掉就可以了

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

insert() 插入

在某个节点之前插入

iterator insert(iterator pos, const T& x)//在摸个节点之前插入
{Node* cur = pos._node;//插入的位置Node* prev = cur->_prev;//插入的下一个位置Node* newnode = new Node(x);//构成节点prev->_next = newnode;newnode->_prev = prev;newnode->_next = cur;cur->_prev = newnode;return newnode;
}

erase() 删除

删除某个节点

iterator erase(iterator pos)
{assert(pos !=end());//判断他不是哨兵位Node* cur = pos._node;Node* prev = cur->_prev;Node* next = cur->_next;prev->_next = next;next->_prev = prev;delete cur;return next;
}

push_back() 尾插

注释这段因为和insert中基本上没区别,所以简单化了

void push_back(const T& x)//尾插
{/*Node* newnode = new Node(x);Node* tail = _head->_prev;//链表尾部节点tail->_next = newnode;newnode->_prev = tail;newnode->_next = _head;_head->_prev = newnode;*/insert(end(), x);
}

push_front() 头插

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

pop_back() 尾删

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

pop_front() 头删

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

begin() 头节点

这里因为是要头节点,所以我们直接把begin设置为第一个节点(有用的节点)

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

end() 尾节点

因为end是尾 所以我们把哨兵位当尾,这样子就可以更好的读

const_iterator end() const
{return _head;
}

总结

可以先尝试一下 自己实现
代码总体加我自己的注释给在这里
实现完可以自己对比一下

template<class T>
//链表节点类
struct listNode
{listNode<T>* _next;listNode<T>* _prev;T date;listNode(const T& x = T()):_next(nullptr),_prev(nullptr),date(x){}
};//正向迭代器类//Ref 来区别const和普通template<class T,class Ref,class Ptr>struct __list__iterator{typedef listNode<T> Node;typedef __list__iterator<T,Ref> self;Node* _node;__list__iterator(Node* node):_node(node){}//++itself& operator++(){_node = _node->_next;return *this;}//it++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->date;}bool operator!=(const self& s){return _node != s._node;}bool operator==(const self& s){return _node == s._node;}Ptr operator->(){return &_node->date;}};
template<class T>
class list
{typedef listNode<T> Node;
public:typedef __list__iterator<T,T&,T*> iterator;typedef __list__iterator<T,const T&,const T*> const_iterator;typedef Reverselterator<T, T&, T*> reverse_iterator;const_iterator begin() const{return _head->_next;}const_iterator end() const{return _head;}iterator begin(){//return iterator(_head->_next);return _head->_next;}iterator end(){//return iterator(_head);return _head;}iterator rbegin(){return end();}iterator rend(){return begin();}void empty_init()//初始化节点{_head = new Node;_head->_next = _head;_head->_prev = _head;}list(){empty_init();}list(list<T>& lt){empty_init();for (const auto& ch : lt){push_back(ch);}}~list(){clear();delete _head;}void clear(){iterator it = begin();while (it != end()){it = erase(it);}}void swap(list<T>& lt){std::swap(_head, lt._head);}list<T>& operator=(list<T> lt){swap(lt);return *this;}void push_back(const T& x)//尾插{/*Node* newnode = new Node(x);Node* tail = _head->_prev;//链表尾部节点tail->_next = newnode;newnode->_prev = tail;newnode->_next = _head;_head->_prev = newnode;*/insert(end(), x);}void push_front(const T& x)//头插{insert(begin(), x);}void pop_back()//尾删{erase(--end());}void pop_front()//头删{erase(begin());}iterator insert(iterator pos, const T& x)//在摸个节点之前插入{Node* cur = pos._node;Node* prev = cur->_prev;Node* newnode = new Node(x);prev->_next = newnode;newnode->_prev = prev;newnode->_next = cur;cur->_prev = newnode;return newnode;}iterator erase(iterator pos){assert(pos !=end());Node* cur = pos._node;Node* prev = cur->_prev;Node* next = cur->_next;prev->_next = next;next->_prev = prev;delete cur;return next;}
private:Node* _head;//哨兵位
};

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

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

相关文章

zabbix监控中间件服务

zabbix监控Nginx 自定义nginx访问量的监控项&#xff0c;首先要通过脚本将各种状态的值取出来&#xff0c;然后通过zabbix监控。找到自定义脚本上传到指定目录/etc/zabbix/script/ 在zbx-client客户端主机操作 #创建目录&#xff0c;然后将脚本上传到该目录mkdir /etc/zabbix/…

C# Dictionary<string, string> 对key做筛选

​ 工作中遇到了一个筛选Dictionary的Key的需求&#xff0c;记录一下。 场景&#xff1a;筛选出所有不包含点&#xff08;.&#xff09;的键。 备选方法&#xff1a; 直接使用for循环遍历&#xff0c;代码量较大。 使用Linq。 最终方法&#xff1a; 选择使用Linq的where…

【JS】关于this的使用

this 前言一、this是什么&#xff1f;二、做什么&#xff1f;1.全局环境2.函数环境3.new实例对象4.apply、bind、call绑定4.1 apply()4.2 call()4.3 bind() 三、为什么用this&#xff1f;四、如何改变this&#xff1f;五、应用场景&#xff1f;总结 前言 痛点 经常写Vue项目&a…

2023 年 3 月青少年软编等考 C 语言一级真题解析

目录 T1. 字符长方形思路分析 T2. 长方形面积思路分析 T3. 成绩等级转换思路分析 T4. 机智的小明思路分析 T5. 寻找特殊年号思路分析 T1. 字符长方形 给定一个字符&#xff0c;用它构造一个长为 4 4 4 个字符&#xff0c;宽为 3 3 3 个字符的长方形&#xff0c;可以参考样例…

C++单例模式、工厂模式

一、单例模式 (一) 什么是单例模式 1. 是什么&#xff1f; 在系统的整个生命周期内&#xff0c;一个类只允许存在一个实例。 2. 为什么&#xff1f; 两个原因&#xff1a; 节省资源。方便控制&#xff0c;在操作公共资源的场景时&#xff0c;避免了多个对象引起的复杂操作…

Unity的相机跟随和第三人称视角二

Unity的相机跟随和第三人称视角二 展示介绍第二种相机模式远离代码总结 展示 我录的视频上可能看不太出来&#xff0c;但是其实今天的这个方法比原来写的那个方法更简便并且死角更少一些。 这里我也做了两个人物偏移的视角&#xff0c;按Q是原来的两个相机模式切换&#xff0c…

论文阅读笔记 | MetaIQA: Deep Meta-learning for No-Reference Image Quality Assessment

文章目录 文章题目发表年限期刊/会议名称论文简要动机主要思想或方法架构实验结果 文章链接&#xff1a;https://doi.org/10.48550/arXiv.2004.05508 文章题目 MetaIQA: Deep Meta-learning for No-Reference Image Quality Assessment 发表年限 2020 期刊/会议名称 Publi…

BSD-3-Clause是一种开源软件许可协议

BSD-3-Clause是一种开源软件许可协议&#xff0c;也称为BSD三条款许可证。它是BSD许可证家族中的一种&#xff0c;是一种宽松的许可证&#xff0c;允许软件自由使用、修改和重新分发&#xff0c;同时也保留了一些版权和责任方面的规定。 BSD-3-Clause许可证的主要特点包括以下…

Marin说PCB之POC电路layout设计仿真案例---01

最近娃哈哈饮料突然爆火&#xff0c;看新闻后才知道春晚的的时候宗老已经病的很严重了&#xff0c;现在也已经离我们而去了&#xff0c;宗老是一个值得我们尊敬爱戴的伟大企业家。于是乎小编我立马去他们的直播间买了一箱娃哈哈AD钙奶支持一下我们的国货。 中午午休的时候&…

vue3中全局变量的定义和获取

在vue项目中&#xff0c;我们知道vue2定义全局变量是在main.js文件将变量挂载到vue.prototype.name"lisi"&#xff0c;在页面通过this.name去调用。但是在vue3中&#xff0c;这个定义全局变量有所改变&#xff1a; const app createApp(App) app.config.globalProp…

Excel如何开启VBA进行二次开发

经常使用Excel做数据分析的朋友平时用的比较多的可能只是一些常用的内置函数或功能&#xff0c;比如求和函数、字符串分割函数、分类汇总、IF函数、VLOOKUP函数等。大多数人认为Excel强大是因为内置了大量的函数。其实&#xff0c;作为一名资深程序猿&#xff0c;个人认为&…

testvue-page

1403.vue <template><div class"error-page"><div class"error-code">4<span>0</span>3</div><div class"error-desc">啊哦~ 你没有权限访问该页面哦</div><div class"error-handle&q…

面试中最常问的10大Linux命令

引言 在信息技术领域中&#xff0c;对于Linux操作系统的精通已成为众多职位的基本要求&#xff0c;尤其对于系统管理员、DevOps工程师、后端开发者等角色更是至关重要。在这篇文章中&#xff0c;我们将深入探讨面试环节中频繁出现的10个核心Linux命令&#xff0c;结合其原理、…

kafka消费端消息去重方案

背景 我们在日常工作中&#xff0c;消费kafka消息是一个最常见的操作&#xff0c;不过由于kafka队列中经常包含重复的消息&#xff0c;并且消息量巨大&#xff0c;所以我们消费端总是需要先把消息进行去重后在消费&#xff0c;以减少消费端的压力&#xff0c;那么日常中我们一…

全量知识系统问题及SmartChat给出的答复 之21 语义关联+词扇+叙词库 之2

Q74. 我们是否可以把语义表征理解为所有语义因子分析的输出结果&#xff0c;换句话说就是是否可以把语义表征作为语义分析的结果的结构来存储分析结果呢 &#xff1f; 是的&#xff0c;语义表征可以被理解为语义分析的结果的结构化表示。语义表征是对语义信息的抽象和总结&…

redis 性能优化一

目录 前言 尾延迟 前言 说到redis 性能优化&#xff0c;优化的目的是什么&#xff1f;提高响应&#xff0c;减少延迟。就要关注两点&#xff0c;一是尾延迟&#xff0c;二是Redis 的基线性能。只有指标&#xff0c;我们的优化&#xff0c;才有意义&#xff0c;才能做监控以及…

玩一会小乌龟

滚滚长江东逝水&#xff0c;浪花淘尽英雄。 是非成败转头空。青山依旧在&#xff0c;几度夕阳红。 白发渔樵江渚上&#xff0c;惯看秋月春风。 一壶浊酒喜相逢。古今多少事&#xff0c;都付笑谈中。 画一个正方形 import turtle# 创建一个Turtle对象 t turtle.Turtle()# 循环…

Spotify8.9.18.512

​ ​【应用名称】&#xff1a;Spotify-声破天 ​【适用平台】&#xff1a;#Android ​【软件标签】&#xff1a;#Spotify ​【应用版本】&#xff1a;8.9.18 ​【应用大小】&#xff1a;67MB ​【软件说明】&#xff1a;软件升级更新。iOS可配合qx小火箭类的工具对该软…

人机环境系统与媒体

人机环境系统是指人与计算机系统或其他数字设备之间的交互环境。它包括硬件设备、软件系统、用户界面和交互设计等组成部分。人机环境系统的目标是提供一个用户友好、高效、安全、可靠的交互环境&#xff0c;使人们能够方便地使用计算机系统或其他数字设备进行工作、学习和娱乐…

scss for循环,$变量,全局样式

$fonWeight: 这段代码是一个 SCSS&#xff08;Sassy CSS&#xff09;代码片段&#xff0c;用于生成一系列字体加粗的类。让我解释一下其中的含义&#xff1a; $fonWeight&#xff1a;这是一个变量&#xff0c;包含了一系列字体加粗的值。它包括数字&#xff08;100 到 900&…