$Date:\ Nov\ 20\ 2020$
Tag: Code
C++ STL容器 学习笔记
P.S.不过与其说是学习,倒不如说是手动换原吧!因为早就学过应用了
[$Vector$]:
优秀的动态数组,支持动态修改大小。
实现原理:
开辟一块连续的内存,不断加入元素。
当内存不足时,申请更大的一块连续内存,复制旧内存区域的元素至新开辟的内存区域,释放旧内存
通常这个扩大的内存大小为原本的1.5倍,保证了不会平凡发生内存二次分配,保证运行效率。
具体维护方法:
记录 _$first$ 指针指向首地址。
记录 $\ \space$ _$last$ 指针指向最后一个元素的下一位。
记录 $\ \space$ _$end$ 指针指向当前申请内存的尾地址的下一位。
故$size() = $ _$last - $ _$first$ , $capacity() = $ _$end - $ _$first$。
P.S. $Hexo$ 渲染$LaTeX$下划线有$bug$,我傻了
核心函数:
1 2
| bool reserve(size_t _count) {}
|
More about reserve
1 2 3 4 5 6 7 8 9 10 11 12 13 14
| bool reserve(size_t _count) { if (capacity() < _count) { _Tp *_ptr = _Buy(_count); if (_ptr == NULL) return (true); if (_Allocate(_ptr, _count)) return (true); if (_first != 0) _Tidy(_first); _end = _ptr + _count; _last = _ptr + size(); _first = _ptr; } return (false); }
|
$reserve$ 实现:
- _$count \le capacity()$ 内存足够,不做任何事。
- _$count \lt capacity()$ 触发二次分配:
- 申请 _$count$ 大小的内存, 记录首指针为 $ ^*$ _$ptr$。
- 复制 _$first \to $ _$last$ 内容至 _$ptr$。
- 清理并释放 _$first \to$ _$end$ 的内存区域。
- 将 _$first$, _$last$, _$end$ 指向新位置。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136
| #include <bits/stdc++.h>
#ifndef _MY_VECTOR_H #define _MY_VECTOR_H 1 template<class _Tp> class vector { static std::string runtime_error; protected: _Tp *_first = 0; _Tp *_last = 0; _Tp *_end = 0; private: static bool _Fill(_Tp *_dest, _Tp *_begin_src, _Tp *_end_src) { for (_Tp *it = _begin_src; it < _end_src; ++it, ++_dest) *(_dest) = *(it); return (false); } static bool _Fill(_Tp *_begin, _Tp *_end, _Tp _init) { for (_Tp *it = _begin; it < _end; ++it) *(it) = _init; return (false); } _Tp* _Buy(size_t _count) { if (!_count) return (NULL); else { _Tp *_ptr = new _Tp[_count]; return _ptr; } } bool _Allocate (_Tp* _ptr, size_t _count) { if (size() > _count) return (true); if (_Fill(_ptr, _first, _last)) return (true); return (false); } bool _Tidy(_Tp *_first) { delete [] _first; return (false); } bool _Destory(_Tp *_begin, _Tp *_end) { for (_Tp *it = _begin; it < _end; ++it) (*it).~_Tp(); return (false); } public: class iterator { _Tp *it; public: iterator(_Tp* _init_it = 0) { it = _init_it; } _Tp& operator* () { if (!it) throw runtime_error; return *it; } iterator& operator++ () { ++it; return *(this); } iterator operator+ (const size_t _plus) { return it + _plus; } iterator operator- (const size_t _dec) { return it - _dec; } bool operator== (const iterator _other) { return it == _other.it; } bool operator!= (const iterator _other) { return it != _other.it; } bool operator< (const iterator _other) { return it < _other.it; } bool operator> (const iterator _other) { return it > _other.it; } bool operator<= (const iterator _other) { return *(this) > _other || *(this) == _other; } bool operator>= (const iterator _other) { return *(this) < _other || *(this) == _other; } }; bool reserve(size_t _count) { if (capacity() < _count) { _Tp *_ptr = _Buy(_count); if (_ptr == NULL) return (true); if (_Allocate(_ptr, _count)) return (true); if (_first != 0) _Tidy(_first); _end = _ptr + _count; _last = _ptr + size(); _first = _ptr; } return (false); } bool resize(size_t _count, _Tp _init = _Tp()) { if (_count == size()) return (false); if (_count < size()) { _Destory(_first + _count, _last); _last = _first + _count; return (false); } else { if (_count > capacity() && reserve(_count)) return (true); _Fill(_last, _first + _count, _init); _last = _first + _count; return (false); } } bool push_back(const _Tp val) { if (size() < capacity()) { *_last = val; _last++; return (false); } else { if (!capacity() && reserve(2)) return true; else if (reserve(capacity() + capacity() / 2)) return (true); *_last = val; _last++; return (false); } } bool pop_back() { if (!size()) { throw runtime_error; return true; } else { _Destory(_last - 1, _last); _last--; return false; } } size_t capacity() { return _end - _first; } size_t size() { return _last - _first; } iterator begin() { return iterator(_first); } iterator end() { return iterator(_last); } _Tp& operator[] (const int& p) { if (p >= size()) throw runtime_error; return *(_first + p); } void clear() { _Destory(_first, _last, _end); if (_first) _Tidy(_first); _first = 0, _last = 0, _end = 0; } vector(size_t _capacity = 0) { reserve(_capacity); } ~vector() { if (_first) _Tidy(_first); } }; template<class _Tp> std::string vector<_Tp>::runtime_error = "runtime_error"; #endif
|
暂时没有好的代码块折叠方法,先这样。。。
先写到这,下次更新随缘
$Nov.\ 20th\ 18:31$