This repository has been archived on 2023-05-13. You can view files and clone it, but cannot push or open issues or pull requests.
mipt_cpp/seminar03_initialization/08_stringview/miptstring.cpp

219 lines
3.9 KiB
C++
Raw Normal View History

2022-10-22 18:26:37 +03:00
#include <iostream>
#include <algorithm>
#include <cstdlib>
#include <cstring>
2022-10-22 22:03:40 +03:00
#include "miptstring.h"
#include "miptstringview.h"
2022-10-22 18:26:37 +03:00
using std::cout, std::cin, std::endl, std::size_t;
namespace mipt{
char* errorCheckedMalloc(size_t newCapacity)
{
char* result = static_cast<char*>(std::malloc(newCapacity * sizeof(char)));
if (result == NULL)
{
cout << "Error! Out of memory" << endl;
std::exit(1);
}
return result;
}
2022-10-22 21:32:50 +03:00
String::String(const char* str)
2022-10-22 18:26:37 +03:00
{
2022-10-22 21:32:50 +03:00
size_t strSize = std::strlen(str);
resize(strSize);
std::memcpy(mpData, str, mSize);
}
2022-10-22 18:26:37 +03:00
2022-10-22 21:32:50 +03:00
String::String() : String("") {}
String::String(const String& s) : String(s.cStr()) {}
String::String(const StringView& sv) {
2022-10-22 19:15:34 +03:00
mSize = sv.size();
2022-10-22 22:03:40 +03:00
(*this).reserve(mSize);
2022-10-22 19:15:34 +03:00
for(int i = 0; i < mSize; ++i)
mpData[i] = sv[i];
mpData[mSize] = '\0';
2022-10-22 22:03:40 +03:00
}
2022-10-22 18:26:37 +03:00
2022-10-22 21:32:50 +03:00
String::String(size_t n, char a)
{
resize(n);
2022-10-22 18:26:37 +03:00
2022-10-22 21:32:50 +03:00
for (size_t i = 0; i < mSize; ++i)
mpData[i] = a;
}
2022-10-22 18:26:37 +03:00
2022-10-22 21:32:50 +03:00
String::~String()
{
std::free(mpData);
}
2022-10-22 18:26:37 +03:00
2022-10-22 21:32:50 +03:00
void String::reserve(size_t capacity)
{
if (capacity <= mCapacity)
return;
2022-10-22 18:26:37 +03:00
2022-10-22 21:32:50 +03:00
mCapacity = std::max(2 * mCapacity, capacity);
char* newData = errorCheckedMalloc(mCapacity);
2022-10-22 18:26:37 +03:00
2022-10-22 21:32:50 +03:00
if (mpData)
std::memcpy(newData, mpData, mSize + 1);
2022-10-22 18:26:37 +03:00
2022-10-22 21:32:50 +03:00
std::free(mpData);
mpData = newData;
}
2022-10-22 18:26:37 +03:00
2022-10-22 21:32:50 +03:00
void String::resize(size_t size)
{
reserve(size + 1);
mSize = size;
mpData[mSize] = '\0';
}
2022-10-22 18:26:37 +03:00
2022-10-22 21:32:50 +03:00
String& String::operator=(const String& right)
{
if (this == &right)
return *this;
2022-10-22 18:26:37 +03:00
2022-10-22 21:32:50 +03:00
mSize = right.mSize;
resize(mSize);
2022-10-22 18:26:37 +03:00
2022-10-22 21:32:50 +03:00
std::memcpy(mpData, right.mpData, mSize + 1);
2022-10-22 18:26:37 +03:00
2022-10-22 21:32:50 +03:00
return *this;
}
2022-10-22 18:26:37 +03:00
2022-10-22 21:32:50 +03:00
String String::operator+(const String& b)
{
String result;
result.resize(mSize + b.mSize);
2022-10-22 18:26:37 +03:00
2022-10-22 21:32:50 +03:00
std::memcpy(result.mpData, mpData, mSize);
std::memcpy(result.mpData + mSize, b.mpData, b.mSize);
result.mpData[result.mSize] = '\0';
2022-10-22 18:26:37 +03:00
2022-10-22 21:32:50 +03:00
return result;
}
2022-10-22 18:26:37 +03:00
2022-10-22 21:32:50 +03:00
String& String::operator+=(const String& right)
{
*this = *this + right;
return *this;
}
2022-10-22 18:26:37 +03:00
2022-10-22 21:32:50 +03:00
bool String::operator==(const String& right) const
{
if (mSize != right.mSize)
return false;
2022-10-22 18:26:37 +03:00
2022-10-22 21:32:50 +03:00
size_t i = 0;
while (i < mSize && mpData[i] == right.mpData[i])
i++;
2022-10-22 18:26:37 +03:00
2022-10-22 21:32:50 +03:00
return i == mSize;
}
2022-10-22 18:26:37 +03:00
2022-10-22 21:32:50 +03:00
bool String::operator<(const String& right) const
{
size_t i = 0;
while (i < mSize && i < right.mSize && mpData[i] == right.mpData[i])
i++;
2022-10-22 18:26:37 +03:00
2022-10-22 21:32:50 +03:00
return mpData[i] < right.mpData[i];
}
2022-10-22 18:26:37 +03:00
2022-10-22 21:32:50 +03:00
bool String::operator<=(const String& right) const
{
size_t i = 0;
while (i < mSize && i < right.mSize && mpData[i] == right.mpData[i])
i++;
2022-10-22 18:26:37 +03:00
2022-10-22 21:32:50 +03:00
return mpData[i] <= right.mpData[i];
}
2022-10-22 18:26:37 +03:00
2022-10-22 21:32:50 +03:00
bool String::operator!=(const String& right) const
{
return !(*this == right);
}
2022-10-22 18:26:37 +03:00
2022-10-22 21:32:50 +03:00
bool String::operator>(const String& right) const
{
return !(*this <= right);
}
2022-10-22 18:26:37 +03:00
2022-10-22 21:32:50 +03:00
bool String::operator>=(const String& right) const
{
return !(*this < right);
}
2022-10-22 18:26:37 +03:00
2022-10-22 21:32:50 +03:00
char& String::operator[](size_t i)
{
return mpData[i];
}
2022-10-22 18:26:37 +03:00
2022-10-22 21:32:50 +03:00
const char& String::operator[](size_t i) const
{
return mpData[i];
}
char& String::at(size_t i)
{
if (i >= mSize)
2022-10-22 18:26:37 +03:00
{
2022-10-22 21:32:50 +03:00
cout << "Error! Index is out of bounds." << endl;
2022-10-22 18:26:37 +03:00
}
2022-10-22 21:32:50 +03:00
return mpData[i];
}
2022-10-22 18:26:37 +03:00
2022-10-22 21:32:50 +03:00
void String::clear()
{
std::free(mpData);
2022-10-22 18:26:37 +03:00
2022-10-22 21:32:50 +03:00
mSize = 0;
mCapacity = 1;
mpData = errorCheckedMalloc(mCapacity);
mpData[0] = '\0';
}
2022-10-22 18:26:37 +03:00
2022-10-22 21:32:50 +03:00
void String::addCharacter(char c)
{
if (mSize + 1 == mCapacity)
reserve(2 * mCapacity);
2022-10-22 18:26:37 +03:00
2022-10-22 21:32:50 +03:00
mpData[mSize] = c;
resize(mSize + 1);
}
2022-10-22 18:26:37 +03:00
2022-10-22 21:32:50 +03:00
size_t String::getSize() const {return mSize;}
size_t String::getCapacity() const {return mCapacity;}
const char* String::cStr() const {return mpData;}
2022-10-22 18:26:37 +03:00
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 (x == ' ' || x == '\n' || x == '\t')
break;
s.addCharacter(x);
}
return in;
}
}