BAN: Rewrite vector with move semantics

This commit is contained in:
Bananymous 2023-01-13 13:50:47 +02:00
parent 065e23d307
commit 0f4e95acc5
1 changed files with 122 additions and 46 deletions

View File

@ -2,13 +2,12 @@
#include <BAN/Errors.h> #include <BAN/Errors.h>
#include <BAN/Math.h> #include <BAN/Math.h>
#include <BAN/Memory.h> #include <BAN/Move.h>
#include <string.h>
namespace BAN namespace BAN
{ {
// T must be move assignable, move constructable (and copy constructable for some functions)
template<typename T> template<typename T>
class Vector class Vector
{ {
@ -18,14 +17,21 @@ namespace BAN
public: public:
Vector() = default; Vector() = default;
Vector(Vector<T>&&);
Vector(const Vector<T>&); Vector(const Vector<T>&);
~Vector(); ~Vector();
Vector<T>& operator=(Vector<T>&&);
Vector<T>& operator=(const Vector<T>&);
[[nodiscard]] ErrorOr<void> PushBack(T&&);
[[nodiscard]] ErrorOr<void> PushBack(const T&); [[nodiscard]] ErrorOr<void> PushBack(const T&);
[[nodiscard]] ErrorOr<void> Insert(T&&, size_type);
[[nodiscard]] ErrorOr<void> Insert(const T&, size_type); [[nodiscard]] ErrorOr<void> Insert(const T&, size_type);
void PopBack(); void PopBack();
void Remove(size_type); void Remove(size_type);
void Clear();
bool Has(const T&) const; bool Has(const T&) const;
@ -42,39 +48,97 @@ namespace BAN
bool Empty() const; bool Empty() const;
size_type Size() const; size_type Size() const;
size_type Capasity() const; size_type Capacity() const;
private: private:
T* Address(size_type, uint8_t* = nullptr) const;
[[nodiscard]] ErrorOr<void> EnsureCapasity(size_type); [[nodiscard]] ErrorOr<void> EnsureCapasity(size_type);
private: private:
T* m_data = nullptr; uint8_t* m_data = nullptr;
size_type m_capasity = 0; size_type m_capacity = 0;
size_type m_size = 0; size_type m_size = 0;
}; };
template<typename T>
Vector<T>::Vector(Vector<T>&& other)
{
m_data = other.m_data;
m_capacity = other.m_capacity;
m_size = other.m_size;
other.m_data = nullptr;
other.m_capacity = 0;
other.m_size = 0;
}
template<typename T> template<typename T>
Vector<T>::Vector(const Vector<T>& other) Vector<T>::Vector(const Vector<T>& other)
{ {
MUST(EnsureCapasity(other.m_size)); MUST(EnsureCapasity(other.m_size));
for (size_type i = 0; i < other.m_size; i++) for (size_type i = 0; i < other.m_size; i++)
m_data[i] = other[i]; new (Address(i)) T(other[i]);
m_size = other.m_size; m_size = other.m_size;
} }
template<typename T> template<typename T>
Vector<T>::~Vector() Vector<T>::~Vector()
{ {
for (size_type i = 0; i < m_size; i++) Clear();
m_data[i].~T(); }
delete[] m_data;
template<typename T>
Vector<T>& Vector<T>::operator=(Vector<T>&& other)
{
Clear();
m_data = other.m_data;
m_capacity = other.m_capacity;
m_size = other.m_size;
other.m_data = nullptr;
other.m_capacity = 0;
other.m_size = 0;
}
template<typename T>
Vector<T>& Vector<T>::operator=(const Vector<T>& other)
{
Clear();
MUST(EnsureCapasity(other.Size()));
for (size_type i = 0; i < other.Size(); i++)
new (Address(i)) T(other[i]);
m_size = other.m_size;
return *this;
}
template<typename T>
ErrorOr<void> Vector<T>::PushBack(T&& value)
{
TRY(EnsureCapasity(m_size + 1));
new (Address(m_size)) T(Move(value));
m_size++;
return {};
} }
template<typename T> template<typename T>
ErrorOr<void> Vector<T>::PushBack(const T& value) ErrorOr<void> Vector<T>::PushBack(const T& value)
{ {
return PushBack(Move(T(value)));
}
template<typename T>
ErrorOr<void> Vector<T>::Insert(T&& value, size_type index)
{
ASSERT(index <= m_size);
TRY(EnsureCapasity(m_size + 1)); TRY(EnsureCapasity(m_size + 1));
m_data[m_size] = value; if (index < m_size)
{
new (Address(m_size)) T(Move(*Address(m_size - 1)));
for (size_type i = m_size - 1; i > index; i--)
*Address(i) = Move(*Address(i - 1));
}
*Address(index) = Move(value);
m_size++; m_size++;
return {}; return {};
} }
@ -82,19 +146,14 @@ namespace BAN
template<typename T> template<typename T>
ErrorOr<void> Vector<T>::Insert(const T& value, size_type index) ErrorOr<void> Vector<T>::Insert(const T& value, size_type index)
{ {
ASSERT(index <= m_size); return Insert(Move(T(value)), index);
TRY(EnsureCapasity(m_size + 1));
memmove(m_data + index + 1, m_data + index, (m_size - index) * sizeof(T));
m_data[index] = value;
m_size++;
return {};
} }
template<typename T> template<typename T>
void Vector<T>::PopBack() void Vector<T>::PopBack()
{ {
ASSERT(m_size > 0); ASSERT(m_size > 0);
m_data[m_size - 1].~T(); Address(m_size - 1)->~T();
m_size--; m_size--;
} }
@ -102,16 +161,28 @@ namespace BAN
void Vector<T>::Remove(size_type index) void Vector<T>::Remove(size_type index)
{ {
ASSERT(index < m_size); ASSERT(index < m_size);
m_data[index].~T(); for (size_type i = index; i < m_size - 1; i++)
memmove(m_data + index, m_data + index + 1, (m_size - index - 1) * sizeof(T)); *Address(i) = Move(*Address(i + 1));
Address(m_size - 1)->~T();
m_size--; m_size--;
} }
template<typename T>
void Vector<T>::Clear()
{
for (size_type i = 0; i < m_size; i++)
Address(i)->~T();
BAN::deallocator(m_data);
m_data = nullptr;
m_capacity = 0;
m_size = 0;
}
template<typename T> template<typename T>
bool Vector<T>::Has(const T& other) const bool Vector<T>::Has(const T& other) const
{ {
for (size_type i = 0; i < m_size; i++) for (size_type i = 0; i < m_size; i++)
if (m_data[i] == other) if (*Address(i) == other)
return true; return true;
return false; return false;
} }
@ -120,59 +191,53 @@ namespace BAN
const T& Vector<T>::operator[](size_type index) const const T& Vector<T>::operator[](size_type index) const
{ {
ASSERT(index < m_size); ASSERT(index < m_size);
return m_data[index]; return *Address(index);
} }
template<typename T> template<typename T>
T& Vector<T>::operator[](size_type index) T& Vector<T>::operator[](size_type index)
{ {
ASSERT(index < m_size); ASSERT(index < m_size);
return m_data[index]; return *Address(index);
} }
template<typename T> template<typename T>
const T& Vector<T>::Back() const const T& Vector<T>::Back() const
{ {
ASSERT(m_size > 0); ASSERT(m_size > 0);
return m_data[m_size - 1]; return *Address(m_size - 1);
} }
template<typename T> template<typename T>
T& Vector<T>::Back() T& Vector<T>::Back()
{ {
ASSERT(m_size > 0); ASSERT(m_size > 0);
return m_data[m_size - 1]; return *Address(m_size - 1);
} }
template<typename T> template<typename T>
const T& Vector<T>::Front() const const T& Vector<T>::Front() const
{ {
ASSERT(m_size > 0); ASSERT(m_size > 0);
return m_data[0]; return *Address(0);
} }
template<typename T> template<typename T>
T& Vector<T>::Front() T& Vector<T>::Front()
{ {
ASSERT(m_size > 0); ASSERT(m_size > 0);
return m_data[0]; return *Address(0);
} }
template<typename T> template<typename T>
ErrorOr<void> Vector<T>::Resize(size_type size) ErrorOr<void> Vector<T>::Resize(size_type size)
{ {
TRY(EnsureCapasity(size));
if (size < m_size) if (size < m_size)
{
for (size_type i = size; i < m_size; i++) for (size_type i = size; i < m_size; i++)
m_data[i].~T(); Address(i)->~T();
m_size = size; if (size > m_size)
}
else if (size > m_size)
{
TRY(EnsureCapasity(size));
for (size_type i = m_size; i < size; i++) for (size_type i = m_size; i < size; i++)
m_data[i] = T(); new (Address(i)) T();
m_size = size;
}
m_size = size; m_size = size;
return {}; return {};
} }
@ -197,26 +262,37 @@ namespace BAN
} }
template<typename T> template<typename T>
typename Vector<T>::size_type Vector<T>::Capasity() const typename Vector<T>::size_type Vector<T>::Capacity() const
{ {
return m_capasity; return m_capacity;
} }
template<typename T> template<typename T>
ErrorOr<void> Vector<T>::EnsureCapasity(size_type size) ErrorOr<void> Vector<T>::EnsureCapasity(size_type size)
{ {
if (m_capasity >= size) if (m_capacity >= size)
return {}; return {};
size_type new_cap = BAN::Math::max<size_type>(size, m_capasity * 3 / 2); size_type new_cap = BAN::Math::max<size_type>(size, m_capacity * 3 / 2);
T* new_data = new T[new_cap]; uint8_t* new_data = (uint8_t*)BAN::allocator(new_cap * sizeof(T));
if (new_data == nullptr) if (new_data == nullptr)
return Error::FromString("Vector: Could not allocate memory"); return Error::FromString("Vector: Could not allocate memory");
if (m_data) for (size_type i = 0; i < m_size; i++)
memcpy(new_data, m_data, m_size * sizeof(T)); {
delete[] m_data; new (Address(i, new_data)) T(Move(*Address(i)));
Address(i)->~T();
}
BAN::deallocator(m_data);
m_data = new_data; m_data = new_data;
m_capasity = new_cap; m_capacity = new_cap;
return {}; return {};
} }
template<typename T>
T* Vector<T>::Address(size_type index, uint8_t* base) const
{
if (base == nullptr)
base = m_data;
return (T*)(base + index * sizeof(T));
}
} }