You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

268 lines
7.7 KiB

#pragma once
#include "iterator.hpp"
2 years ago
#include "reverse_iterator.hpp"
2 years ago
#include "utils.hpp"
#include <iostream>
#include <memory>
2 years ago
#include <stdexcept>
namespace ft {
2 years ago
template < class T, class Alloc = std::allocator< T > > class vector {
public:
2 years ago
typedef T value_type;
typedef Alloc allocator_type;
typedef typename Alloc::pointer pointer;
typedef typename Alloc::const_pointer const_pointer;
typedef value_type &reference;
typedef const value_type &const_reference;
typedef ft::random_access_iterator< T > iterator;
typedef ft::random_access_iterator< const T > const_iterator;
typedef ft::reverse_iterator< iterator > reverse_iterator;
typedef ft::reverse_iterator< const_iterator > const_reverse_iterator;
typedef std::ptrdiff_t difference_type;
typedef std::size_t size_type;
private:
2 years ago
Alloc _alloc;
pointer _begin;
size_type _size;
size_type _capacity;
public:
2 years ago
explicit vector(const Alloc &alloc = allocator_type())
: _alloc(alloc), _begin(NULL), _size(0), _capacity(0){};
2 years ago
explicit vector(size_type count, const_reference value = T(),
const Alloc &alloc = allocator_type())
: _alloc(alloc), _size(count), _capacity(count) {
_begin = _alloc.allocate(count);
for (size_type i = 0; i < count; i++)
2 years ago
_alloc.construct(_begin + i, value);
}
template < class It >
2 years ago
vector(It first, It last, const Alloc &alloc = allocator_type(),
2 years ago
typename ft::enable_if< !is_integral< It >::value, bool >::type = 0)
: _alloc(alloc) {
2 years ago
_size = ft::distance(first, last) - 1;
_capacity = _size;
_begin = _alloc.allocate(_size);
for (size_type i = 0; i < _size; i++)
2 years ago
_alloc.construct(_begin + i, *(first++));
}
vector(const vector &other)
: _alloc(other._alloc), _size(other._size), _capacity(other._capacity) {
_begin = _alloc.allocate(_capacity);
for (size_type i = 0; i < _size; i++)
2 years ago
_alloc.construct(_begin + i, other._begin[i]);
}
~vector(void) {
clear();
if (_capacity)
_alloc.deallocate(_begin, _capacity);
}
2 years ago
vector &operator=(const vector &other) {
clear();
_alloc.deallocate(_begin, _capacity);
_alloc = other._alloc;
_capacity = other._capacity;
_size = other._size;
_begin = _alloc.allocate(_capacity);
for (size_type i = 0; i < _size; i++)
2 years ago
_alloc.construct(_begin + i, other._begin[i]);
2 years ago
return (*this);
}
2 years ago
void assign(size_type count, const_reference value) {
2 years ago
resize(count);
2 years ago
while (count--) {
2 years ago
_alloc.destroy(_begin + count);
_alloc.construct(_begin + count, value);
2 years ago
};
2 years ago
}
2 years ago
template < class It >
2 years ago
void assign(It first, It last,
2 years ago
typename ft::enable_if< !is_integral< It >::value, bool >::type = 0) {
2 years ago
size_type count = ft::distance(first, last) - 1;
resize(count);
2 years ago
size_t i = 0;
while (i < count) {
_alloc.destroy(_begin + i);
_alloc.construct(_begin + i, *(first++));
i++;
2 years ago
}
2 years ago
}
2 years ago
allocator_type get_allocator(void) const { return _alloc; }
// ACCESS:
2 years ago
reference at(size_type pos) {
2 years ago
if (pos >= _size)
throw(std::out_of_range("vector: Out of range access"));
2 years ago
return *(_begin + pos);
}
2 years ago
const_reference at(size_type pos) const {
2 years ago
if (pos >= _size)
throw(std::out_of_range("vector: Out of range access"));
2 years ago
return *(_begin + pos);
}
2 years ago
reference operator[](size_type pos) { return _begin[pos]; }
const_reference operator[](size_type pos) const { return _begin[pos]; }
2 years ago
reference front(void) { return _begin[0]; }
const_reference front(void) const { return _begin[0]; }
2 years ago
reference back(void) { return _begin[_size - 1]; }
const_reference back(void) const { return _begin[_size - 1]; }
2 years ago
pointer data(void) { return _begin; }
const_pointer data(void) const { return _begin; }
// ITERATORS:
2 years ago
iterator begin(void) { return iterator(_begin); }
const_iterator begin(void) const { return const_iterator(_begin); }
2 years ago
iterator end(void) { return iterator(_begin + _size); }
const_iterator end(void) const { return const_iterator(_begin + _size); }
2 years ago
reverse_iterator rbegin(void) { return reverse_iterator(end()); }
const_reverse_iterator rbegin(void) const {
return const_reverse_iterator(end());
2 years ago
}
2 years ago
reverse_iterator rend(void) { return reverse_iterator(begin()); }
const_reverse_iterator rend(void) const {
return const_reverse_iterator(begin());
2 years ago
}
// CAPACITY:
2 years ago
bool empty(void) const { return _size == 0; }
size_type size(void) const { return _size; }
size_type max_size(void) const { return _alloc.max_size(); }
void reserve(size_type new_cap) {
if (new_cap > max_size())
throw(std::length_error("vector::reserve: new_cap > max_size"));
2 years ago
if (new_cap < _capacity)
return ;
2 years ago
Alloc new_alloc;
pointer new_begin;
2 years ago
new_begin = new_alloc.allocate(new_cap);
2 years ago
size_t new_size;
new_size = new_cap < _capacity ? new_cap : _size;
for (size_type i = 0; i < new_size; i++)
2 years ago
new_alloc.construct(new_begin + i, *(_begin + i));
2 years ago
if (!empty())
clear();
if (_capacity)
_alloc.deallocate(_begin, _capacity);
2 years ago
_alloc = new_alloc;
_begin = new_begin;
_capacity = new_cap;
2 years ago
_size = new_size;
2 years ago
}
size_type capacity(void) const { return _capacity; }
// MODIFIERS:
void clear(void) {
for (size_type i = 0; i < _size; i++)
_alloc.destroy(_begin + i);
_size = 0;
}
2 years ago
iterator insert(const_iterator pos, const_reference value) {
2 years ago
if (_size == _capacity)
resize(_size + 1);
2 years ago
iterator it = end();
while (--it > pos)
2 years ago
*it = *(it - 1);
*it = value;
return _begin;
}
2 years ago
iterator insert(const_iterator pos, size_type count, const_reference value) {
2 years ago
if ((_size + count) > _capacity)
2 years ago
resize(_size + count);
2 years ago
iterator it = end();
2 years ago
while (--it >= (pos + count))
*it = *(it - count);
2 years ago
while (it >= pos)
*(it--) = value;
return it;
2 years ago
}
2 years ago
template < class It > iterator insert(const_iterator pos, It first, It last,
typename ft::enable_if< !is_integral< It >::value, bool >::type = 0) {
2 years ago
size_type count = ft::distance(first, last) - 1;
2 years ago
resize(_size + count);
2 years ago
iterator it = end();
2 years ago
while (--it > (pos + count))
2 years ago
*it = *(it - count);
2 years ago
while (it > pos) {
*(it--) = *(last--);
2 years ago
}
2 years ago
return it;
2 years ago
}
2 years ago
2 years ago
iterator erase(iterator pos) {
2 years ago
while (pos < end()) {
2 years ago
_alloc.destroy(pos.base());
2 years ago
*pos = *(pos + 1);
pos++;
}
_size--;
return _begin;
}
2 years ago
iterator erase(iterator first, iterator last) {
while (last++ < end()) {
2 years ago
_alloc.destroy(first.base());
2 years ago
*(first++) = *last;
}
while (first < end() && _size--) {
2 years ago
_alloc.destroy(first.base());
2 years ago
*(first++) = value_type();
}
return _begin;
}
2 years ago
2 years ago
void push_back(const_reference value) {
2 years ago
if (_size == _capacity)
resize(_size + 1);
2 years ago
_alloc.construct(_begin + _size - 1, value);
2 years ago
}
2 years ago
void pop_back(void) {
2 years ago
_alloc.destroy(_begin + _size - 1);
2 years ago
_size--;
}
2 years ago
void resize(size_type count, T value = T()) {
2 years ago
if (count < _size) {
2 years ago
while (count < --_size)
_alloc.destroy(_begin + _size);
2 years ago
} else {
if (count > _capacity)
reserve(count);
2 years ago
while (_size < count)
_alloc.construct(_begin + _size++, value);
2 years ago
}
}
2 years ago
2 years ago
void swap(vector &other) {
2 years ago
vector< T > tmp(other);
2 years ago
other = this;
this = tmp;
}
2 years ago
void print(void) {
2 years ago
ft::random_access_iterator< T > i = begin();
2 years ago
while (i < end()) {
std::cout << *i << " ";
i++;
}
std::cout << "\n";
}
};
} // namespace ft