C++初阶:适合新手的手撕list(模拟实现list)

这篇具有很好参考价值的文章主要介绍了C++初阶:适合新手的手撕list(模拟实现list)。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

上次讲了常用的接口:今天就来进行模拟实现啦



1.基本结构与文件规划

C++初阶:适合新手的手撕list(模拟实现list),c++学习,c++,list,windows,java,linux,tcp/ip

  • list.h头文件:包含类的全部(函数的声明与定义)
  • reverseIterator.h文件:进行反向迭代器的实现
  • test.cpp源文件:进行调用test函数,测试和完善功能

基本结构:

#pragma once

namespace MyList
{
    // List的节点类
    template<class T>
    struct ListNode
    {
        ListNode<T>* _prev;
        ListNode<T>* _next;
        T _data;

        ListNode(const T& x=T())
            :_prev(nullptr)
            ,_next(nullptr)
            ,_data(x)
        {}
    };


    //List的迭代器类
    template<class T, class Ref, class Ptr>
    struct ListIterator
    {
        typedef ListNode<T> Node;
        typedef ListIterator<T, Ref, Ptr> Self;
        
        Node* _node;

        ListIterator(Node* node)//构造函数
            :_node(node)
        {}

        ListIterator(const Self& l)//拷贝构造函数
            :_node(l._node)
        {}

        T& operator*();
        T* operator->();
        Self& operator++();
        Self operator++(int);
        Self& operator--();
        Self& operator--(int);
        bool operator!=(const Self& l);
        bool operator==(const Self& l);

    };


    //list类
    template<class T>
    class list
    {
        typedef ListNode<T> Node;//默认是private 不给外面用
    public:
        typedef ListIterator<T, T&, T*> iterator;
        typedef ListIterator<T, const T&, const T&> const_iterator;
        //构造函数
        list();

        list(int n, const T& value = T());

        template <class Iterator>
        list(Iterator first, Iterator last);

        //析构
        ~list();


        // List Iterator
        iterator begin();
        iterator end();
        const_iterator begin();
        const_iterator end();


        // List Capacity
        size_t size()const;
        bool empty()const;


   
        // List Access
        T& front();
        const T& front()const;
        T& back();
        const T& back()const;


        // List Modify
        void push_back(const T& val) { insert(end(), val); }
        void pop_back() { erase(--end()); }
        void push_front(const T& val) { insert(begin(), val); }
        void pop_front() { erase(begin()); }
        // 在pos位置前插入值为val的节点
        iterator insert(iterator pos, const T& val);
        // 删除pos位置的节点,返回该节点的下一个位置
        iterator erase(iterator pos);
        void clear();
        void swap(list<T>& l);

    private:
        Node* _head;
    };
};
  1. ListNode 结构体:
    • 定义了链表的节点结构,包含了三个成员变量:前驱指针 _prev、后继指针 _next 和数据 _data
    • 构造函数初始化了这些成员变量,允许在创建节点时指定初始值。
  2. ListIterator 结构体:
    • 定义了链表的迭代器结构,包含了指向节点的指针 _node
    • 重载了一系列操作符,如 *->++--!===,以便于对链表进行遍历和操作。
  3. list 类:
    • 包含了迭代器的定义、构造函数、析构函数以及一系列的操作函数。
    • 定义了两种迭代器类型:iteratorconst_iterator,分别用于可修改的迭代和只读的迭代。
    • 实现了一系列的操作函数

2.空参构造函数(constructor)

        list()
        {
            _head = new Node;//去调用Node的默认构造函数了
            _head->_next = _head;
            _head->_prev = _head;//带头双向循环链表是这样的
        }

使用new:动态开辟+调用默认构造函数了


3.完善迭代器(iterator)(begin(),end())

这里为什么我们要把迭代器封装为一个类呢?明明之前模拟vectorstring时,就直接typedef

之前模拟vectorstring时,二者底层都是连续的,想要到下一个可以直接++;想要得到里面的数据可以直接*

但是现在对于list是不行的,我们就需要重载各种运算符,但是底层又是一个指针(内置类型)不能重载,现在就只能封装进一个类里,就能重载

 //List的迭代器类
    template<class T, class Ref, class Ptr>
    struct ListIterator
    {
        typedef ListNode<T> Node;
        typedef ListIterator<T, Ref, Ptr> Self;
        
        Node* _node;

        ListIterator(Node* node)//构造函数
            :_node(node)
        {}

        ListIterator(const Self& l)//拷贝构造函数
            :_node(l._node)//这里是浅拷贝(写不写都行)
            //新创建的迭代器和原迭代器指向相同的内存地址
        {}

        Ref operator*()
        {
            return _node->_data;
        }
        Ptr operator->()
        {
            return &(_node->_data);
        }

        Self& operator++()//前置
        {
            _node = _node->_next;//自己要更新
            return *this;
        }
        Self operator++(int)
        {
            Self s(*this);
            _node = _node->_next;
            return s;
        }

        Self& operator--()
        {
            _node = _node->_prev;//自己要更新
            return *this;
        }
        Self& operator--(int)
        {
            Self s(*this);
            _node = _node->_prev;
            return s;
        }

        bool operator!=(const Self& l)
        {
            return _node != l._node;
        }
        bool operator==(const Self& l)
        {
            return _node == l._node;
        }

    };


    //list类
    template<class T>
    class list
    {
        typedef ListNode<T> Node;//默认是private 不给外面用
    public:
        typedef ListIterator<T, T&, T*> iterator;
        typedef ListIterator<T, const T&, const T&> const_iterator;
        //构造函数
        list()
        {
            _head = new Node;//去调用Node的默认构造函数了
            _head->_next = _head;
            _head->_prev = _head;//带头双向循环链表是这样的
        }

        // List Iterator
        iterator begin()
        {
            return _head->_next;//隐式类型转换(由单参构造函数支撑)
        }
        iterator end()
        {
            return _head;
        }
        const_iterator begin()
        {
            return _head->_next;
        }
        const_iterator end()
        {
            return _head;
        }

    private:
        Node* _head;
    };
};

4.List Capacity(size(),empty())

       // List Capacity
        size_t size()const
        {
            size_t size = 0;
            iterator it = begin();
            while (it != end())
            {
                size++;
                ++it;
            }
            return size;
        }
        bool empty()const
        {
            return size() == 0;
        }

4.增删改查(push_back,pop_back,pop_front,push_front,insert,erase)

        // List Modify
        void push_back(const T& val) //尾插
        { 
            insert(end(), val); 
        }
        void pop_back() //尾删
        { 
            erase(--end());
        }
        void push_front(const T& val) //头插
        { 
            insert(begin(), val);
        }
        void pop_front()//头删
        { 
            erase(begin());
        }
        // 在pos位置前插入值为val的节点
        iterator insert(iterator pos, const T& val)
        {
            Node* cur = pos._node;
            Node* prev = pos._node->_prev;
            Node* newnode = new Node(val);//创建新节点

            newnode->_next = cur;
            cur->_prev = newnode;
            newnode->_prev = prev;
            prev->_next = cur;

            return newnode;//隐式类型转换
        }
        // 删除pos位置的节点,返回该节点的下一个位置
        iterator erase(iterator pos)
        {
            assert(pos != _head);

            Node* cur = pos._node;
            Node* prev = cur->_prev;
            Node* next = cur->_next;

            prev->_next = next;
            next->_prev = prev;

            delete cur;
            return next;
        }

使用test1函数看功能是否正常

	void print(MyList::list<int>& lt)
	{
		list<int>::iterator it = lt.begin();
		while (it != lt.end())
		{
			cout << *it << " ";
			++it; // 更新迭代器
		}
		cout << endl;
	}

	void test1()
	{
		MyList::list<int> lt;
		lt.push_back(1);
		lt.push_back(2);//尾插2个
		print(lt);

		lt.pop_back();//尾删一个
		print(lt);

		lt.push_front(0);//头插一个
		print(lt);

		lt.pop_front();//头删一个
		print(lt);
	}

C++初阶:适合新手的手撕list(模拟实现list),c++学习,c++,list,windows,java,linux,tcp/ip


6.clear()和swap()

       void clear()
        {
            //删除除头结点(_head)以外的节点
            iterator it = begin();
            while (it != end())
            {
                it = erase(it);
            }
        }

        void swap(list<T>& l)
        {
            std::swap(_head, l._head);
        }

7. 完善构造函数

7.1list (size_type n, const value_type& val = value_type());

list(int n, const T& value = T())
{
    _head = new Node;
    _head->_next = _head;
    _head->_prev = _head;

    for (int i = 0; i < n; ++i)
    {
        push_back(value);
    }
}

C++初阶:适合新手的手撕list(模拟实现list),c++学习,c++,list,windows,java,linux,tcp/ip

7.2利用迭代器进行构造

	    template <class Iterator>
        list(Iterator first, Iterator last)
        {
            while (first != last)
            {
                push_back(*first);
                first++;
            }
        }

为什么使用模版:

因为可能使用其他类型的迭代器来进行初始化

7.3拷贝构造

        list(const list<T>& lt)
        {
            _head = new Node;
            _head->_next = _head;
            _head->_prev = _head;

            iterator it = copy.begin();
            while (it != copy.end())
            {
                push_back(*it);
                it++;
            }
        }

8.重载=

        list<T>& lt operator=(list<T> lt)
        {
            swap(lt);
            return *this;
        }

注意这里的参数不是常量引用,而是按值传递的。这是因为在赋值操作符中我们会调用 swap 函数,按值传递可以保证传入的参数会被复制一份,避免对原对象的修改。在函数体内,我们调用了 swap 函数,将当前对象和传入的对象进行内容交换,然后返回 *this,即当前对象的引用。


9.析构函数

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

调用clear函数后,就只剩下头结点了


10.反向迭代器

我们再次使用封装的思想,封装一个反向迭代器进去

#pragma once

template <class iterator,class Ref,class Pre>
struct reserve_iterator
{
	typedef reserve_iterator<iterator, Ref, Pre> self;

	iterator _it;

	reserve_iterator(iterator it)
		:_it(it)  
	{}

	self& operator++()
	{
		--_it;
		return *this;
	}
	self operator++(int)
	{
		self tmp = *this;
		--_it;
		return tmp;
	}

	self& operator--()
	{
		++_it;
		return *this;
	}
	self operator--(int)
	{
		self tmp = *this;
		++_it;
		return tmp;
	}

	Ref operator*()
	{
		iterator tmp = _it;
		--tmp;
		return *tmp;
	}

	Pre operator->()
	{
		return &(operator*());
	}

	bool operator!=(const self& s)
	{
		return _it != s._it;
	}
	bool operator==(const self& s)
	{
		return _it == s._it;
	}

};

此时那list类里就是这样:

C++初阶:适合新手的手撕list(模拟实现list),c++学习,c++,list,windows,java,linux,tcp/ip


好啦,list的内容也结束啦,下次就是Stack和Queue了。感谢大家支持!!!文章来源地址https://www.toymoban.com/news/detail-830523.html

到了这里,关于C++初阶:适合新手的手撕list(模拟实现list)的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • C++初阶之一篇文章教会你list(模拟实现)

    成员类型表 这个表中列出了C++标准库中list容器的一些成员类型定义。这些类型定义是为了使list能够与C++标准库的其他组件协同工作,并提供一些通用的标准接口。每个成员类型的用处: value_type : 这个成员类型代表list容器中存储的数据类型,即模板参数T的类型。 allocator_

    2024年02月12日
    浏览(26)
  • 【C++】手撕string(string的模拟实现)

    手撕string目录: 一、 Member functions 1.1 constructor 1.2  Copy constructor(代码重构:传统写法和现代写法) 1.3 operator=(代码重构:现代写法超级牛逼) 1.4 destructor 二、Other member functions 2.1 Iterators(在string类中,迭代器基本上就是指针) 2.1.1 begin() end() 2.1.2  范围for的底层

    2024年02月08日
    浏览(34)
  • 【C++初阶9-list实现】封装——我主沉浮,何不能至?

    翻阅文档即可。 带头双向循环链表。 *数据结构初阶已经实现过,这里只是变成类和对象的版本 list的实现,价值最大的不是这里,而是迭代器…… 封装,我主沉浮,何不能至? 封装给了我们很灵活的操作空间。为什么这么说? 迭代器是一种类的内嵌类型,具有指针行为,

    2023年04月27日
    浏览(27)
  • C++初阶-vector类的模拟实现

    C++ STL中的vector就类似于C语言当中的数组,但是vector又拥有很多数组没有的接口,使用起来更加方便。 相比于STL中的string,vector可以定义不同的数据类型。 迭代器的本质可以暂时看作是指针,模拟实现vector,需要定义三个指针:指向起始位置_start,指向最后一个有效元素的下

    2024年02月04日
    浏览(145)
  • 【C++初阶】9. string类的模拟实现

    string类的完整实现放这里啦!快来看看吧 string类的作用就是将字符串类型实现更多功能,运算符重载,增删改查等等操作,所以其成员就包含char*的字符串 在之前的学习过程中,我们了解到类中存在的六个默认函数,其中就包含默认构造函数,那么对于string类是否需要用户自

    2024年02月09日
    浏览(27)
  • 【C++初阶】学习string类的模拟实现

    前面已经学习了string类的用法,这篇文章将更深入的学习string类,了解string类的底层是怎么实现的。当然,这里只是模拟一些常用的,不常用的可以看文档学习。 我们一共创建两个文件,一个是test.cpp文件,用于测试;另一个是string.h文件,用于声明和定义要模拟的string类。

    2024年02月03日
    浏览(36)
  • 【C++初阶】模拟实现string的常见操作

    👦个人主页:@Weraphael ✍🏻作者简介:目前学习C++和算法 ✈️专栏:C++航路 🐋 希望大家多多支持,咱一起进步!😁 如果文章对你有帮助的话 欢迎 评论💬 点赞👍🏻 收藏 📂 加关注✨ 为了方便管理代码,分两个文件来写: Test.cpp - 测试代码逻辑 string.h - 模拟实现 strin

    2024年02月12日
    浏览(38)
  • 【C++模拟实现】list的模拟实现

    作者:爱写代码的刚子 时间:2023.9.3 前言:本篇博客关于list的模拟实现和模拟实现中遇到的问题 list模拟实现的部分代码 list模拟实现中的要点 const_iterator的实现 我们选择使用模版参数,复用iterator的类,设置三个模版参数: templateclass T,class Ref,class Ptr 并且 typedef __list_iter

    2024年02月09日
    浏览(38)
  • 【C++初阶】10. vector的使用及模拟实现

    vector的文档介绍 vector是表示可变大小数组的序列容器。 就像数组一样,vector也采用的连续存储空间来存储元素。也就是意味着可以采用下标对vector的元素进行访问,和数组一样高效。但是又不像数组,它的大小是可以动态改变的,而且它的大小会被容器自动处理。 必要时查

    2024年02月06日
    浏览(38)
  • 【C++初阶】第八站:string类的模拟实现

    目录 string类的模拟实现 经典的string类问题 浅拷贝 深拷贝 写时拷贝(了解) 构造函数 string的全缺省的构造函数: string的拷贝构造函数 传统写法 现代写法 string的赋值重载函数 传统写法 现代写法 string的无参构造函数: 遍历函数 operator[ ] 迭代器 迭代器的底层实现begin和end:

    2024年04月28日
    浏览(28)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包