-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathE39.h
165 lines (143 loc) · 3.71 KB
/
E39.h
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
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
#ifndef E39_H
#define E39_H
#include <string>
using std::string;
#include <memory>
using std::allocator;
#include <utility>
using std::pair;
#include <iostream>
using std::endl;
using std::cout;
#include <initializer_list>
using std::initializer_list;
#include <algorithm>
class StrVec{
public:
StrVec(): element(nullptr), first_free(nullptr), cap(nullptr) {};
StrVec(initializer_list<string> il);
StrVec(const StrVec&);
StrVec& operator=(const StrVec&);
~StrVec();
string* begin() const { return element; }
string* end() const { return first_free; }
void push_back(const string& s);
size_t size() const { return first_free - element; }
size_t capacity() const { return cap - element; }
void reserve(size_t n);
void resize(size_t n);
void resize(size_t n, const string& s);
private:
static allocator<string> alloc;
string* element;
string* first_free;
string* cap;
void free(); // free the all_element in the vector and then free the vector
void reallocate(); // when the storage is not enough, reallocate the new memory
void chk_n_alloc(); // check if there is one more storage for the new element
pair<string*, string*> alloc_n_copy(const string* b, const string* e); // copy from the old vector for construct and assignment
void alloc_n_move(size_t newcap);
};
allocator<string> StrVec:: alloc = allocator<string>(); // static member must be defined outside the function
StrVec::StrVec(initializer_list<string> il){
/*
size_t newcap = il.size();
auto newdata = alloc.allocate(newcap);
auto dest = newdata;
for(auto begin = il.begin(); begin != il.end(); ++begin){
alloc.construct(dest++, *begin);
}
element = newdata;
first_free = dest;
cap = first_free;
*/
auto newdata = alloc_n_copy(il.begin(), il.end());
element = newdata.first;
first_free = newdata.second;
cap = first_free;
}
/*
void StrVec::free(){
if(element){
for(auto p = first_free; p != element;){
alloc.destroy(--p);
}
alloc.deallocate(element, cap-element);
}
}
*/
void StrVec::free(){
if(element){
for_each(begin(), end(), [](const string& s){ alloc.destroy(&s); } );
alloc.deallocate(element, cap-element);
}
}
void StrVec::chk_n_alloc(){
if (first_free == cap) {
reallocate();
cout << "memory is not enough!" << endl;
}
}
void StrVec::alloc_n_move(size_t newcap){ // we can also call the alloc_n_copy function here
auto newdata = alloc.allocate(newcap);
auto dest = newdata;
auto ele = element;
for(size_t i = 0; i < size(); ++i){
alloc.construct(dest++, std::move(*ele++));
}
free();
element = newdata;
first_free = dest;
cap = element + newcap;
}
void StrVec::reallocate(){
auto newcapacity = size() ? 2*size() : 1;
alloc_n_move(newcapacity);
}
pair<string*, string*> StrVec::alloc_n_copy(const string* b, const string* e){
auto newv = alloc.allocate(e-b);
auto ret = uninitialized_copy(b, e, newv);
return {newv, ret};
}
StrVec::StrVec(const StrVec& sv){
auto new_sv = alloc_n_copy(sv.begin(), sv.end());
element = new_sv.first;
first_free = new_sv.second;
cap = first_free;
}
StrVec& StrVec::operator=(const StrVec& sv){
auto new_sv = alloc_n_copy(sv.begin(), sv.end());
free();
element = new_sv.first;
first_free = new_sv.second;
cap = first_free;
return *this;
}
StrVec::~StrVec(){
free();
}
void StrVec::push_back(const string& s){
chk_n_alloc();
alloc.construct(first_free, s);
++first_free;
}
void StrVec::reserve(size_t n){
if(n <= capacity()) return;
alloc_n_move(n);
}
void StrVec::resize(size_t n){
resize(n, string());
}
void StrVec::resize(size_t n, const string& s){
size_t raw_size = size();
if(n < raw_size){
while(first_free != element + n){
alloc.destroy(--first_free);
}
} else if(n > raw_size){
for(size_t j = raw_size; j < n; ++j){
push_back(s);
}
}
}
#endif