1.首先要了解到vs底层的list链表是带头双向循环的链表。
所以首先就要看成员变量
data:image/s3,"s3://crabby-images/7d4bd/7d4bd82af189f7c225369ff570a9f36c1c03d7ac" alt=""
那么就说明我们还需要构造一个Node的结构体,(typedef一下就好了,名字不影响)
data:image/s3,"s3://crabby-images/3b926/3b926423549a20bd408275b45983ba4b99eb4be9" alt=""
现在就可以完成间的push_back函数了。
data:image/s3,"s3://crabby-images/9cbd1/9cbd19980d0e257b99978e04c315feb748009e17" alt=""
1.list的iterator
我们之前模拟vector和string的时候就直接typedef T* iterator
data:image/s3,"s3://crabby-images/037b8/037b823e372dd71630e834b81cd35ed6d71b9dc6" alt=""
但是这个list是双向的迭代器,这个可以在C++的官网上可以查询的。
data:image/s3,"s3://crabby-images/1f300/1f300c2c4902fb659cc670ac774ed32bd8792e9b" alt=""
其他类型也是可以查询的:
data:image/s3,"s3://crabby-images/7ce01/7ce01f075f8eb559ceebc322a9498bde1f415481" alt=""
因为链表的指针注定是不可以简单的++就可以完成的,vector是用数组存储的,所以指针都是连续的,所以我们就必须想别的方法,C++牛逼的就是如果我们把iterator也封装成一个类/结构体,那么我们就可以干我们想的任何事情,因为他可以运算符重载,所以我们外边看着和vector一样的++,但是底层是完全不同的。这才是C++祖师爷强的地方,外面都是vector<>::iterator/list<>::iterator, 但是底层是天差地别。
data:image/s3,"s3://crabby-images/69a3d/69a3dce912a2905df5a22e374c9e8e25390717e3" alt=""
2.list的const_iterator
有些人的想法是typedef const _list_iterator<T> const_iterator,但是使用的时候,Q::list<int>::const_iterator it = l.begin(); ------> 相当于 Q::list<int>:: const _list_iterator it = l.begin()
那么_list_iterator 这个类是不发改变的,但是const迭代器期望指向内容是不能修改的,这样设计的迭代器本身不能修改。必然是有_node = _node->_next。所以这不能和vector一样无脑的typedef的。
更有有说那么久重新设计一个_const_list_iterator的类就好了嘛。但是那就太过冗余了,因为里面只有部分的成员函数不一样,所以我们就不需要再设计一个类了,我们先看看他们两者类有什么区别,才能更好的统一。
data:image/s3,"s3://crabby-images/06b0a/06b0abcd5b877c9d7faf0c127b93b1a4e917da0b" alt=""
这里只有返回值是不一样的,所以为什么不能将返回值再用一个模板来统一规划一下呢?
data:image/s3,"s3://crabby-images/68ce4/68ce46042eafaf3e66bea20ec0f959494e95bb03" alt=""
所以就变成了:
data:image/s3,"s3://crabby-images/60fc4/60fc43691548af09c65b86c3c561f15c90f3816f" alt=""
3._list_iterator类的再次优化
data:image/s3,"s3://crabby-images/0948c/0948caf283638b8e6d7d4119c945e80109806162" alt=""
看一眼这个就知道要重载运算符的-> , 那就实现呗。
data:image/s3,"s3://crabby-images/6e1c1/6e1c18ea31d0cc261c362c94d2c30ff4ed577e36" alt=""
data:image/s3,"s3://crabby-images/c0020/c0020f2fa4cb2a2099768b19ae86ccdb5a049c47" alt=""
但是他实现的很奇怪:
但是如果是const类型怎么办?因为它的返回值是const T*, 所以我们就有必要再加一个类模板参数了: