list就是我们之前学过的链表,我们都知道链表和string、vector不一样,链表自身是不连续的一段内存,而string和vector是连续的。并且这里的list是双向带头的链表。因此我们先来看链表的节点:
template<class T>
struct list_node
{list_node(const T& val = T()):next(nullptr),prev(nullptr),data(val){}list_node<T>* next;list_node<T>* prev;T data;
};
我们使用模板来表示链表的一个节点,这一个节点里面有一个T类型的数据,并且有两个方向next和prev,两个节点的指针来存储另外两个节点的信息。
对于容器,我们都需要有迭代器,像我们使用过的string 和 vector 都有迭代器,我们在模拟基础实现的时候,直接使用了原生指针typedef成了迭代器,但是对于链表来说,它的内存是不连续的,因此不能进行随机访问,也就是不能进行+,+=这些操作,因此我们必须封装一个迭代器:
template<class T,class Ref, class Ptr>
struct list_iterator
{typedef list_node<T> node;typedef list_iterator<T,Ref,Ptr> self;list_iterator(node* val):_node(val){}Ref operator*(){return _node->data;}Ptr operator->(){return &_node->data;}self& operator++(){_node = _node->next;return *this;}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;}bool operator!=(const self& val) const{return _node != val._node;}bool operator==(const self& val) const{return _node == val._node;}node* _node;
};
这里我们使用了模板来完成链表的迭代器,对于iterator 和 const_iterator,区别就是能不能对此处位置值的修改,因此我们完全可以使用模板来完成两个迭代器。
T这个参数就是数据类型,也就是我们链表里面存的是什么类型的数据。对于Ref和Ptr,其实就是在迭代器那里,如果我们传入的是const_iterator,那么Ref,Ptr也就会被编译器推断成相应正确的类型。
Ref operator*()
{return _node->data;
}Ptr operator->()
{return &_node->data;
}
迭代器的本质其实也就是一个节点,只不过是对这个节点进行了封装,我们在使用迭代器时,就可以通过迭代器来操作相应的节点位置。并且我们对迭代器也进行了相应的运算符重载,例如前置、后置++(--),!= 和 ==,解引用* ,->等操作,这些操作都比较简单,返回对应的结果即可,唯一要注意的是,->运算符重载时,是返回节点数据的地址。
下面我们来看看list类的实现:
class list
{
public:typedef list_node<T> node;typedef list_iterator<T,T&,T*> iterator;typedef list_iterator<T, const T&, const T*> const_iterator;private:node* _head;size_t _size;};
我们先来看看三个typedef:
- typedef list_node<T> node;这个比较好理解,我们在创建一个节点时,需要使用到 list_node<T>,因为太长所以直接定义为node。
- typedef list_iterator<T,T&,T*> iterator;
- typedef list_iterator<T, const T&, const T*> const_iterator;
这里就能明白,为什么我们在封装迭代器的时候需要传入三个模板参数,当我们使用不同的迭代器时,传入的参数就不一样,就可以满足普通迭代器和const迭代器的区别了。
list内部成员就只有一个头节点和 _size实时记录list的大小。
我们先来看list的一些构造函数:
void Init_Empty()
{_head = new node;_head->next = _head;_head->prev = _head;_size = 0;
}list()
{Init_Empty();
}list(size_t n, const T& val = T())
{Init_Empty();for (size_t i = 0; i < n; ++i){push_back(val);}
}list(int n, const T& val = T())
{Init_Empty();for (int i = 0; i < n; ++i){push_back(val);}
}template <class Iterator>
list(Iterator first, Iterator last)
{Init_Empty();while (first != last){push_back(*first);++first;}
}list(const list<T>& lt)
{Init_Empty();for (auto e : lt)push_back(e);
}void swap(list<T> lt)
{std::swap(_head, lt._head);std::swap(_size, lt._size);
}list<T>& operator=(const list<T> lt)
{swap(lt);return *this;
}~list()
{clear();delete _head;_head = nullptr;
}void clear()
{auto it = begin();while (it != end()){it = erase(it);}
}
我们先实现了一个Init_Empty函数,这个函数就是帮助我们先构造处一个头节点,因此list无参构造就可以直接调用这个函数。
接下来是用n个val值初始化list:
list(size_t n, const T& val = T())
{Init_Empty();for (size_t i = 0; i < n; ++i){push_back(val);}
}list(int n, const T& val = T())
{Init_Empty();for (int i = 0; i < n; ++i){push_back(val);}
}
因为我们的头节点是不算在链表数据之内的,它只是一个帮助我们找到其他节点的一个工具。所以我们先调用 Init_Empty,完成头节点的构造,再向头节点后面不断插入即可。
template <class Iterator>
list(Iterator first, Iterator last)
{Init_Empty();while (first != last){push_back(*first);++first;}
}list(const list<T>& lt)
{Init_Empty();for (auto e : lt)push_back(e);
}
迭代器区间构造和拷贝构造函数也比较简单,思想和n个val初始化类似,这里不做过多解释。
void swap(list<T> lt)
{std::swap(_head, lt._head);std::swap(_size, lt._size);
}list<T>& operator=(const list<T> lt)
{swap(lt);return *this;
}~list()
{clear();delete _head;_head = nullptr;
}
void clear()
{auto it = begin();while (it != end()){it = erase(it);}
}
最后是赋值重载和析构,对于赋值重载,我们只需要交换对应list的成员变量即可,和前面的string vector思想一致。对于析构函数,我们直接调用了clear(),因为clear的功能就是从begin()开始,一直删除直到剩下头节点,然后再delete头节点即可。
接下来是迭代器,这里比较简单,也就不解释了:
iterator begin()
{return _head->next;
}iterator end()
{return _head;
}const_iterator begin()const
{return _head->next;
}const_iterator end()const
{return _head;
}
接下来是insert和erase函数使用:
iterator insert(iterator pos, const T& val)
{node* tmp = new node;node* cur = pos._node;node* prev = cur->prev;tmp->data = val;prev->next = tmp;tmp->prev = prev;tmp->next = cur;cur->prev = tmp;_size++;return tmp;
}
iterator erase(iterator pos)
{assert(pos != end());node* cur = pos._node;node* prev = cur->prev;node* next = cur->next;delete[] cur;cur = nullptr;prev->next = next;next->prev = prev;_size--;return next;
}
我们之前学习链表的时候,对于增删已经非常熟悉,无非就是增加或者删除节点,然后将对应的节点的next与prev调整好指向方向。这里对于insert,因为我们要在pos位置增加一个节点,我们只需要找到pos位置之前的节点prev,然后将新开出来的节点插入到pos和 prev之间即可。
对于erase,我们需要找到pos位置的prev和next,然后delete[ ]掉pos位置的节点,然后将prev和next链接起来就行了,需要注意的只是删除的时候pos位置不能等于end(),因为我们不能删除头节点。
然后是头插尾插和头删尾删:
void push_back(const T& val)
{insert(end(), val);
}void push_front(const T& val)
{insert(begin(), val);
}void pop_back()
{erase(--end());
}void pop_front()
{erase(begin());
}
这里就是insert和erase的复用,找好对应的pos位置即可。
最后就是几个简单的函数功能:
//链表的长度
size_t size()
{return _size;
}//是否为空
bool empty()
{return _size == 0;
}//返回链表第一个数据
T& front()
{iterator it = begin();return it._node->data;
}//返回链表第一个数据
const T& front()const
{const_iterator it = begin();return it._node->data;
}//返回链表最后一个数据
T& back()
{iterator it = end();return it._node->prev->data;
}//返回链表最后一个数据
const T& back()const
{const_iterator it = end();return it._node->prev->data;
}
上面的函数功能都比较好理解,大家知道是干什么的就行,实现起来肯定没问题。
以上内容就是对list的基础功能实现,如有错误,欢迎批评指正!!!