【C++教程从0到1入门编程】第十三篇:STL中list类的模拟实现

一、list的模拟实现

#include<iostream>
#include<assert.h>
#pragma once
namespace jyr
{template<class T>struct _list_node{_list_node<T>* _next;_list_node<T>* _prev;T _data;_list_node(const T& val = T()):_next(nullptr), _prev(nullptr), _data(val){}};template<class T,class Ref,class Ptr>struct _list_iterator{typedef _list_node<T> Node;typedef _list_iterator<T, Ref, Ptr> Self;Node* _node;_list_iterator(Node* node):_node(node){}//*it,返回_dataRef operator*(){return _node->_data;}//it->datePtr operator->(){return &_node->_data;}//it++,返回还是一个迭代器,前置加加Self& operator++(){_node = _node->_next;return *this;}//前置--Self& operator--(){_node = _node->_prev;return *this;}//后置加加Self operator++(int){Self tmp(*this);  //创建临时变量返回,后置加加,加加之前不变//_list_iterator<T>tmp = *this;_node = _node->_next;//++(*this)return tmp;}//后置减减Self operator--(int){Self tmp(*this);_node = _node->_prev;return tmp;}//==bool operator ==(const Self& it){return _node = it._node;}//it!=end()bool operator!=(const Self& it){return _node != it._node;}};template<class T>class list{public:typedef _list_node<T> Node;public:typedef _list_iterator<T,T&,T*> iterator;typedef _list_iterator<T,const T&,const T*> const_iterator;iterator begin()            //第一个数据的迭代器,节点的指针构建迭代器{return iterator(_head->_next);}iterator end()     {return iterator(_head);}const_iterator begin() const{return const_iterator(_head->_next);}const_iterator end() const{return const_iterator(_head);}//带头双向循环链表list()                      //list的构造{_head = new Node;_head->_next = _head;_head->_prev = _head;} //l2(l1)list(const list<T>& l1)            //拷贝构造{//链表的最初状态_head = new Node;_head->_next = _head;_head->_prev = _head;//const_iterator it = l1.begin();   //构造一个const迭代器尾插到新的迭代器中//while (it != l1.end())//{//	this->push_back(*it);//	it++;//}for (auto e : l1){push_back(e);}}list<T>& operator=(const list<T>& l1){if (this != &l1){clear();for (auto e : l1){push_back(e);}}return *this;}//现代版的拷贝构造//list<T>& operator=(list<T> l1)//{//	swap(_head, l1._head);//	return *this;//}void clear(){iterator it = begin();while (it != end()){erase(it++);         //erase要自己单独实现出来}}~list()                      //析构函数{clear();                 //clear没有删除头结点delete _head;_head = nullptr;}size_t size()  const{}bool empty() const{return _head->_next == _head->_prev;}//T x  如果T类型是vector或者是string就造成了深拷贝,那么就得传引用,传引用不改变x就加constvoid push_back(const T& x)  {//Node* tail = _head->_prev;//Node* newnode = new Node(x);//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 insert(iterator pos, const T& x)    //在pos这个迭代器的前面插入一个节点{Node* newnode = new Node(x);//pos 是个迭代器,而迭代器又是节点的指针,所以可以通过迭代器找到这个节点Node* cur = pos._node;Node* prev = cur->_prev;prev->_next = newnode;newnode->_prev = prev;newnode->_next = cur;cur->_prev = newnode;//return iterator(newnode);}void pop_back(){erase(--end());}void pop_front(){erase(begin());//end是最后一个有效数据的下一个位置,所以end先减减,删除end减减的位置}void erase(iterator pos){//pos是一个迭代器,节点的指针,可以找到这个节点//找到要删除的节点delassert(pos != end());//头结点不能删除Node* del = pos._node;Node* next = del->_next;//假设有1 2 3节点,要删除pos位置2节点,//先2节点的前一个节点1节点的下一个节点指向2指向的3节点//然后要把3和1链接起来,把3的前一个节点指向1,//2的下一个节点3节点的前一个节点指向2的前一个节点//删除2del->_prev->_next = del->_next; del->_next->_prev = del->_prev;delete del;//return iterator(next);//返回节点的下一个位置}private:Node* _head;};void testlist1(){list<int> lt1;lt1.push_back(1);lt1.push_back(2);lt1.push_back(10);lt1.push_back(20);lt1.pop_back();list<int>::iterator it = lt1.begin();while (it != lt1.end()){std::cout << *it << " ";++it;}std::cout << std::endl;}struct Date{int _year = 0;int _month = 1;int _day = 1;};void print_list(const list<int>& lt1){list<int>::const_iterator it = lt1.begin();while (it != lt1.end()){std::cout << *it << " ";it++;}std::cout << std::endl;}void testlist2(){list<Date> lt1;lt1.push_back(Date());lt1.push_back(Date());list<Date>::iterator it = lt1.begin();while (it != lt1.end()){/*std::cout << *it << " ";*///it->Date->_year  std::cout << it->_year << ":" << it->_month << ":" << it->_day << std::endl;std::cout << (*it)._year << ":" << (*it)._month << ":" << (*it)._day << std::endl;++it;}//print_list(lt1);}void testlist3(){list<int> l1;l1.push_back(1);l1.push_back(2);l1.push_back(3);l1.push_back(4);list<int>l2(l1); //拷贝构造,用l1拷贝构造l2for (auto e : l1){std::cout << e << " ";}std::cout << std::endl;for (auto e : l2){std::cout << e << " ";}std::cout << std::endl;}
}

二、list与vector之间的对比

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

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

相关文章

Zabbix 配置使用

目录 配置流程 添加组机组 添加模板 添加主机 配置图形 配置大屏 Monitoring 配置地图 最新数据 故障 使用IT服务 使用报表 资产管理 全局搜索 导入导出 用户权限 用户组权限 用户 匿名用户 调试模式 与 LDAP 对接 维护模式 故障确认 批量更新 配置流程…

Spring高频面试题

&#xff08;一些来源于GitCode AI&#xff09; 什么是Spring框架&#xff1f; Spring是一个开源的Java平台&#xff0c;它简化了企业级应用的开发。它提供了IOC&#xff08;Inversion of Control&#xff09;/DI&#xff08;Dependency Injection&#xff09;容器&#xff0c;…

ssh登录qemu虚拟机里的linux系统

1、安装openssh&#xff0c;如果是根文件系统用buildroot构建&#xff0c;打开 BR2_PACKAGE_OPENSSH 开关 2、在qemu的启动脚本里增加 -net user,hostfwdtcp::2222-:22上面的命令启动了一个带有NAT网络的QEMU虚拟机&#xff0c;并设置了端口转发&#xff0c;将主机的2222端口…

【办公类-21-10】三级育婴师 视频转文字docx(等线小五单倍行距),批量改成“宋体小四、1.5倍行距、蓝色字体、去掉五分钟”

作品展示 背景需求 今天将最后3个育婴师操作视频做整理 第1步&#xff1a;视频MP4转MP3 【办公类-40-01】20240311 用Python将MP4转MP3提取音频 &#xff08;家长会系列一&#xff09;-CSDN博客文章浏览阅读393次&#xff0c;点赞9次&#xff0c;收藏6次。【办公类-40-01】20…

day31|leetcode|C++|贪心|455.分发饼干|376. 摆动序列|53. 最大子数组和

局部最优解&#xff0c;是一种感觉 455.分发饼干 链接&#xff1a;455. 分发饼干 thought&#xff1a; 如何让更多的孩子吃到合适的饼干呢&#xff0c;重排来实现局部最优&#xff0c;每次我们都挑出当前胃口最小的孩子&#xff0c;从小到大找饼干&#xff0c;只要满足就res…

Unity关于实现雨刮器shader效果实现

Unity关于实现雨刮器shader效果实现 一、绘制雨滴 二、序列图使用 三、雨滴UV序列图UV相加

打造个性化日期选择:闭包实现的datePicker封装新体验

在数字时代的浪潮中,我们每天都在与各种应用程序和工具打交道。其中,日期选择器(datePicker)无疑是用户界面设计中的一大亮点,它帮助我们快速、准确地选择日期,极大地提升了用户体验。然而,市面上的datePicker组件往往千篇一律,缺乏个性化定制。今天,我要为大家种草一…

前端理论总结(js)——js垃圾回收机制 // 堆和栈的区别

两种方式&#xff1a; 引用计数 1&#xff1a;先声明一个变量&#xff0c;并将一个引用类型的值赋给该变量&#xff0c;那么这个引用类型的引用次数为1&#xff0c;计数为1 2&#xff1a;如果同一个引用类型的值又赋给其他变量&#xff0c;那么这个引用类型的值被引用的次数就…

[项目前置]websocket协议

websocket协议介绍 WebSocket 协议是一种在单个 TCP 连接上进行全双工通讯的协议。 WebSocket 使得客户端和服务器之间的数据交换变得更简单&#xff0c;允许服务器主动向客户端推送数据。它在 2011 年成为国际标准&#xff0c;现在被所有现代浏览器支持。WebSocket 设计用于…

0058__developer-roadmap最全的开发者技术学习路线

GitHub - kamranahmedse/developer-roadmap: Interactive roadmaps, guides and other educational content to help developers grow in their careers. 259k Star&#xff01;这是我见过最全的开发者技术学习路线&#xff01; - 掘金

iNet Network Scanner Mac 网络扫描工具

iNet Network Scanner for Mac是一款功能强大的网络扫描工具&#xff0c;专为Mac用户设计。它提供了全面而深入的网络分析功能&#xff0c;使用户能够轻松获取Mac连接的网络和设备的详细信息。 软件下载&#xff1a;iNet Network Scanner Mac v3.1.0激活版 这款软件具备多种扫描…

WPF —— Menu数据绑定实例

{Binding} 因为我们操作这个集合对象&#xff0c;而不是集中某个对象&#xff0c;所以直接写{Binding}就行 如果绑定是list集合的某个对象属性时候&#xff0c;需要{bindingvpath 属性名} <Menu x:Name"m1" ItemsSource"{Binding}">&l…

业务服务:xss攻击

文章目录 前言一、使用注解预防1. 添加依赖2. 自定义注解3. 自定义校验逻辑4. 使用 二、使用过滤器1. 添加配置2. 创建配置类3. 创建过滤器4. 创建过滤器类5. 使用 前言 xss攻击时安全领域中非常常见的一种方法&#xff0c;保证我们的系统安全是非常重要的 xss攻击简单来说就…

P2123皇后游戏

P2123皇后游戏 参考题解 #include <iostream> #include <algorithm> using namespace std;int T; int n; long long res;struct Person {int a,b,d; }p[20005];bool person_cmp(const Person& x,const Person& y) {if(x.d y.d){if(x.d < 0)return x.a …

torchrun在验证集使用一个GPU时报NCCL超时:Watchdog caught collective operation timeout的解决方案

有时候在分布式训练时&#xff0c;可能会出现nccl通信超时的问题&#xff0c;出现的原因好像是在某些数据处理、加载等过程&#xff0c;多个进程一起完成&#xff0c;但是某些计算&#xff08;比如loss具体不知道都有啥&#xff09;需要rank0自己来做&#xff0c;但是由于默认的…

语言模型:解密AI语言理解之道

在这个信息爆炸的时代,我们每天都在与海量的文本数据打交道。如何从这些文本中提炼出有价值的信息,成为了科技界的一大挑战。而语言模型,正是解决这一挑战的关键技术之一。今天,就让我们一起走进语言模型的神秘世界,探索其原理、实战与评估的奥秘。 一、原理篇:语言模型…

Javascript——Symbol简单了解

一、Symbol(符号) 1.1 MDN链接&#xff1a;yield - JavaScript | MDN (mozilla.org) 1.2 Symbol简介 Symbol(符号)是ECMAScript6新增的数据类型。符号是原始值&#xff0c;且符号实例是唯一、不可变的。符号的用途是确保对象属性使用唯一标识符&#xff0c;不会发生属性冲突…

TCP(socket 套接字)编程 1

一、TCP套接字编程架构如下 二、相关代码实现 1、服务器端代码 package com.company;import java.io.IOException; import java.net.InetSocketAddress; import java.net.ServerSocket; import java.net.Socket;public class Main {public static void main(String[] args) {…

5.88 BCC工具之tcpsynbl.py解读

一,工具简介 tcpsynbl工具以直方图的形式显示SYN到达时的TCP SYN积压大小。这可以让我们了解应用程序距离达到积压限制并丢弃SYN(导致SYN重传产生性能问题)还有多远。 TCP SYN 数据包则通常用于启动 TCP 三次握手过程的第一次握手。 二,代码示例 #!/usr/bin/env python…