You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
This repo is archived. You can view files and clone it, but cannot push or open issues/pull-requests.

215 lines
4.3 KiB
C++

#pragma once
#include <iostream>
#include <cstring>
namespace mipt{
class String
{
private:
std::size_t mSize {0};
std::size_t mCapacity {0};
char* mpData {nullptr};
public:
String(const char* str)
{
std::size_t strSize = std::strlen(str);
resize(strSize);
std::memcpy(mpData, str, mSize);
std::cout << "mipt::String Constructor (" << mpData << ")" << std::endl;
}
String() : String("") {}
String(const String& s) : String(s.cStr()) {}
String(std::size_t n, char a)
{
resize(n);
for (std::size_t i = 0; i < mSize; ++i)
mpData[i] = a;
std::cout << "mipt::String Constructor (" << mpData << ")" << std::endl;
}
~String()
{
std::cout << "mipt::String Destructor (" << mpData << ")" << std::endl;
delete [] mpData;
}
void reserve(std::size_t capacity)
{
if (capacity <= mCapacity)
return;
mCapacity = std::max(2 * mCapacity, capacity);
char* newData = new char[mCapacity]; // errorCheckedMalloc(mCapacity);
if (mpData)
std::memcpy(newData, mpData, mSize + 1);
delete [] mpData;
mpData = newData;
}
void resize(std::size_t size)
{
reserve(size + 1);
mSize = size;
mpData[mSize] = '\0';
}
String& operator=(const String& right)
{
if (this == &right)
return *this;
mSize = right.mSize;
resize(mSize);
std::memcpy(mpData, right.mpData, mSize + 1);
return *this;
}
String operator+(const String& b)
{
String result;
result.resize(mSize + b.mSize);
std::memcpy(result.mpData, mpData, mSize);
std::memcpy(result.mpData + mSize, b.mpData, b.mSize);
result.mpData[result.mSize] = '\0';
return result;
}
String& operator+=(const String& right)
{
*this = *this + right;
return *this;
}
bool operator==(const String& right) const
{
if (mSize != right.mSize)
return false;
std::size_t i = 0;
while (i < mSize && mpData[i] == right.mpData[i])
i++;
return i == mSize;
}
bool operator<(const String& right) const
{
std::size_t i = 0;
while (i < mSize && i < right.mSize && mpData[i] == right.mpData[i])
i++;
return mpData[i] < right.mpData[i];
}
bool operator<=(const String& right) const
{
std::size_t i = 0;
while (i < mSize && i < right.mSize && mpData[i] == right.mpData[i])
i++;
return mpData[i] <= right.mpData[i];
}
bool operator!=(const String& right) const
{
return !(*this == right);
}
bool operator>(const String& right) const
{
return !(*this <= right);
}
bool operator>=(const String& right) const
{
return !(*this < right);
}
char& operator[](std::size_t i)
{
return mpData[i];
}
const char& operator[](std::size_t i) const
{
return mpData[i];
}
char& at(std::size_t i)
{
if (i >= mSize)
throw std::out_of_range{"mipt::String::at: index >= this->size()"};
return mpData[i];
}
const char& at(std::size_t i) const
{
if (i >= mSize)
throw std::out_of_range{"mipt::String::at: index >= this->size()"};
return mpData[i];
}
void clear()
{
delete [] mpData;
mSize = 0;
mCapacity = 1;
mpData = new char[mCapacity];
mpData[0] = '\0';
}
void addCharacter(char c)
{
if (mSize + 1 == mCapacity)
reserve(2 * mCapacity);
mpData[mSize] = c;
resize(mSize + 1);
}
std::size_t getSize() const {return mSize;}
std::size_t getCapacity() const {return mCapacity;}
const char* cStr() const {return mpData;}
};
std::ostream& operator<<(std::ostream& out, const String& s)
{
out << s.cStr();
return out;
}
std::istream& operator>>(std::istream& in, String& s)
{
s.clear();
while (true)
{
char x = in.get();
if (std::isspace(x))
break;
s.addCharacter(x);
}
return in;
}
}