模拟实现简单string

 string作为STL一员在实际中使用非常广泛,我们需要对其常用接口进行掌握和熟练使用。那么对于string的模拟实现也是很有必要的,有助于帮助我们更加了解string的底层实现。

以下为实现:

//string.h文件
#define _CRT_SECURE_NO_WARNINGS 1
#include<iostream>
#include<string.h>
#include<assert.h>
using namespace std;

namespace cls
{
	class string
	{
	public:
		typedef char* iterator;
		typedef const char* const_iterator;

		iterator begin()
		{
			return _str;
		}

		iterator end()
		{
			return _str + _size;
		}

		const_iterator begin() const
		{
			return _str;
		}

		const_iterator end() const
		{
			return _str + _size;
		}

		string(const char* str = "")
		{
			_size = strlen(str);
			_capacity = _size;
			_str = new char[_size+1];
			strcpy(_str,str);
		}

		/*string(const string& s)
		{
			_size = s._size;
			_capacity = s._capacity;
			_str = new char[_capacity + 1];
			strcpy(_str,s._str);
		}*/

		string(const string& s)
		{
			string temp(s._str);
			swap(temp);
		}

		~string()
		{
			delete[] _str;
			_str = nullptr;
			_size = _capacity = 0;
		}

		size_t size() const
		{
			return _size;
		}

		size_t capacity() const
		{
			return _capacity;
		}
		
		char& operator[](size_t pos)
		{
			assert(pos < _size);
			return _str[pos];
		}

		const char& operator[](size_t pos) const
		{
			assert(pos < _size);
			return _str[pos];
		}


		void clear()
		{
			_size = 0;
			_str[0] = '\0';
		}

		const char* c_str() const
		{
			return _str;
		}

		/*string& operator=(const string& s)
		{
			delete[] _str;
			_size = s._size;
			_capacity = s._capacity;
			_str = new char[1 + _capacity];
			strcpy(_str,s._str);

			return *this;
		}*/

		string& operator=(string s)
		{
			swap(s);
			   
			return *this;
		}

		void swap(string& str)
		{
			std::swap(_str, str._str);
			std::swap(_size,str._size);
			std::swap(_capacity,str._capacity);
		}

		void reserve(size_t n);

		void push_back(char ch);

		void append(const char* s);
	
		string& operator+=(char ch);

		string& operator+=(const  char* s);

		void insert(size_t pos,char ch);

		void insert(size_t pos,const char* s);

		void erase(size_t pos , size_t n = npos);

		size_t find(char ch,size_t pos = 0);

		size_t find(const char* s, size_t pos = 0);

		string substr(size_t pos = 0,size_t n = npos);

		bool operator<(const string& str);

		bool operator<=(const string& str);

		bool operator>(const string& str);

		bool operator>=(const string& str);

		bool operator==(const string& str);

		bool operator!=(const string& str);
		
		static const size_t npos;
	private:
		char* _str = nullptr;
		size_t _size = 0;
		size_t _capacity = 0;
	};
	
	ostream& operator<<(ostream& out, const string& str);

	istream& operator>>(istream& in, string& str);

}

//string.cpp文件
#include"string.h"

namespace cls
{
	void string::reserve(size_t n)
	{
		if (n > _capacity)
		{
			char* temp = new char[n+1];
			strcpy(temp,_str);
			_capacity = n;
			delete[] _str;
			_str = temp;
		}
	}

	void  string::push_back(char ch)
	{
		if (_size == _capacity)
		{
			size_t newcapacity = 0 == _capacity ? 4 : 2 * _capacity;
			reserve(newcapacity);
		}
		_str[_size] = ch;
		_str[_size+1] = '\0';
		++_size;
	}

	void string::append(const char* s)
	{
		size_t len = strlen(s);
		if (_size + len > _capacity)
		{
			reserve(_size+len > _capacity*2 ? _size + len : 2*_capacity);
		}
		strcpy(_str+_size,s);
		_size += len;
	}

	string& string::operator+=(char ch)
	{
		push_back(ch);
		return *this;
	}

	string& string::operator+=(const  char* s)
	{
		append(s);
		return *this;
	}

	void string::insert(size_t pos, char ch)
	{
		assert(pos < _size);
		if (_size == _capacity)
		{
			reserve(2*_capacity);
		}
		size_t end = _size + 1;
		while (end > pos)
		{
			_str[end] = _str[end - 1];
			--end;
		}
		_str[pos] = ch;
		++_size;
	}

	void string::insert(size_t pos, const char* s)
	{
		assert(pos < _size);
		size_t len = strlen(s);
		if (_size+len > _capacity)
		{
			reserve(_size+len > 2*_capacity ? _size+len : 2*_capacity);
		}
		size_t end = _size + len;
		while (end > pos +len -1)
		{
			_str[end] = _str[end - len];
			--end;
		}
		size_t i = 0;
		for (i = 0;i < len;++i)
		{
			_str[pos+i] = s[i];
		}
		_size += len;
	}

	void string::erase(size_t pos, size_t n)
	{
		assert(pos <_size);
		if (_size-pos < n)
		{
			n = _size - pos;
		}
		size_t begin = pos + n;
		while (begin <= _size)
		{
			_str[begin - n] = _str[begin];
			++begin;
		}
		_size -= n;
	}

	size_t string::find(char ch, size_t pos)
	{
		assert(pos < _size);
		size_t i = 0;
		for (i = 0; i < _size; ++i)
		{
			if (ch == _str[i])
			{
				return i;
			}
		}
		return npos;
	}

	size_t  string::find(const char* s, size_t pos)
	{
		assert(pos < _size);
		char* ret = strstr(_str+pos,s);
		if (nullptr == ret)
		{
			return npos;
		}
		else
		{
			return ret - _str;
		}
	}

	string string::substr(size_t pos, size_t n)
	{
		assert(pos < _size);
		if (_size - pos < n)
		{
			n = _size - pos;
		}
		string temp;
		reserve(n);
		size_t i = pos;
		for (i = pos; i < pos + n; ++i)
		{
			temp += _str[i];
		}
		return temp;
	}

	bool string::operator<(const string& str)
	{
		return (strcmp(_str,str.c_str()) < 0);
	}

	bool string::operator<=(const string& str)
	{
		return *this < str || *this == str;
	}

	bool string::operator>(const string& str)
	{
		return !(*this <= str);
	}

	bool string::operator>=(const string& str)
	{
		return !(*this < str);
	}

	bool string::operator==(const string& str)
	{
		return strcmp(_str,str.c_str()) == 0;
	}

	bool string::operator!=(const string& str)
	{
		return !(*this == str);
	}

	const size_t string::npos = -1;

	ostream& operator<<(ostream& out, const string& str)
	{
		for (auto ch : str)
		{
			out << ch;
		}
		return out;
	}

	istream& operator>>(istream& in, string& str)
	{
		str.clear();
		const int N = 256;
		char buff[N] = {0};
		char ch = in.get();
		int i = 0;
		while (ch != ' ' && ch != '\n')
		{
			buff[i++] = ch;
			ch = in.get();
			if (N-1 == i)
			{
				buff[i] = '\0';
				str += buff;
				i = 0;
			}
		}
		buff[i] = '\0';
		str += buff;

		return in;
	}

}

以上是直接基于顺序表实现的string的常用接口。希望对大家有所帮助。

  • 5
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值