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的基础功能实现,如有错误,欢迎批评指正!!!