C++学习——string类的模拟实现

news2025/1/22 15:05:05

目录

  string类的创建

  string类的构造函数

  itertor迭代器

  []操作符重载

  字符串修改函数

   1.尾插函数

   2.append函数

   3.+=运算符重载函数

   4.clear函数

   5.swap函数

  容量检测或修改函数

  resize函数

  reserve函数

  经过上一次的博客之后我们已经认识了string类,并且可以使用string类当中的相关的成员函数。为了巩固我们学习到的相关的知识,在本次的博客当中我们将自主实现一个string类来完成相应的功能。

  string类的创建

  首先我们想要创建string类就应该清楚他的存储的原理。实质上就跟我们的顺序表完全相同,申请一段连续的空间,进行数据的存储。所以我们在创建string类之前应该先定义三个变量。它们分别是:反应字符串长度的size,反应空间容量的capacity,为存储数据所开辟的空间str。这三个变量将会是我们string类作用的全部的私有变量。所示代码如下:

#include<iostream>
using namespace std;
//定义一个命名空间于,在其中创建一个字符串类
namespace test
{
    class string
    {
     private:
		int _size;
		int _capacity;
		char* _str;
    };
}

  string类的构造函数

  在设计完所需要的私有成员变量之后,第二步需要做的就是设计构造函数,用于我们类对象的初始化操作。我们可以通过new操作符向堆区申请一段可以使用的空间。并将其他参数进行适当的初始化。

  由于构造函数有很多重载类型,所以我们可以先设计一个简单一点的。比如将一个字符作为参数生成一个字符串对象。代码如下:

#include<iostream>
using namespace std;
//定义一个命名空间于,在其中创建一个字符串类
namespace test
{
    class string
    {
     private:
		int _size;
		int _capacity;
		char* _str;
     public:
        //使用函数重载,丰富构造函数,使我们可以使用多种类型进行对象的初始化
		string(const char ch)
		{
			_size = 1;
			_capacity = 1;
			_str = new char[2];
			_str[0] = ch;
			_str[1] = '\0';
		}
    };
}

   当我们只使用一个字符初始化对象的时候,我们只需要申请两个字符的空间即可,一个字节存放着我们的有效字符,一个字节存放着 ‘ \0 ’ 之后再将我们的 size 和 capacity 均设为1即可。需要注意的是:在设置capacity的时候我们并不将 ‘ \0 ’ 的长度计算在容量当中,所以 capacity 为1。

  之后为了是我们的string类更加丰富,所以我们可以采用函数重载的形式,设计多个构造函数用于,初始化我们的对象。代码如下:

#include<iostream>
using namespace std;
//定义一个命名空间于,在其中创建一个字符串类
namespace test
{
    class string
    {
     private:
		int _size;
		int _capacity;
		char* _str;
     public:
        //使用函数重载,丰富构造函数,使我们可以使用多种类型进行对象的初始化
		string(const char ch)
		{
			_size = 1;
			_capacity = 1;
			_str = new char[2];
			_str[0] = ch;
			_str[1] = '\0';
		}
        //写一个构造函数,用于初始化string类对象
		string(const char*ch="")
		{
			_size = strlen(ch);
			_capacity = _size;
			_str = new char[_capacity+1];
			memcpy(_str, ch, _size+1);
		}
    };
}

  第二个构造函数我们使用一个字符串来初始话我们的对象。我们需要首先需要计算出字符串的长度,之后开辟出指定的空间,最后将字符串当中的数据复制到我们开辟好的空间当中即可。许哟啊注意的是:在复制的时候需要使用memcpy进行复制。因为对于我们的string类对象来说,所存储的数据可能存在 ‘ \0 ’ 如果使用strcpy及进行拷贝就可能造成数据缺失的弊端。

  另一个十分重要的部分就是我们的拷贝构造。拷贝构造作为一种特殊的构造函数我们在初始话构造函数的同时也将其设计出来。代码如下:

#include<iostream>
using namespace std;
//定义一个命名空间于,在其中创建一个字符串类
namespace test
{
    class string
    {
     private:
		int _size;
		int _capacity;
		char* _str;
     public:
        //使用函数重载,丰富构造函数,使我们可以使用多种类型进行对象的初始化
		string(const char ch)
		{
			_size = 1;
			_capacity = 1;
			_str = new char[2];
			_str[0] = ch;
			_str[1] = '\0';
		}
        //写一个构造函数,用于初始化string类对象
		string(const char*ch="")
		{
			_size = strlen(ch);
			_capacity = _size;
			_str = new char[_capacity+1];
			memcpy(_str, ch, _size+1);
		}
        //定义一个拷贝构造
		string(const string& s)
		{
			_size = s._size;
			_capacity = s._capacity;
			_str = new char[s._capacity + 1];
			memcpy(_str, s._str, _size + 1);
		}
    };
}

  在拷贝构造当中我们需要开辟一个和字符串对象相同的空间,之后将字符串对象的所有的特征都赋值给我们创造出来的新的对象即可。

上面三种构造函数是string类生成对象最常用的三个构造函数。我们先构建出最常用的功能以小见大即可。

  同样的,说到构造函数自然而然就会想到我们的析构函数,在析构函数的部分当中我们要处理的功能为:释放我们开辟的堆区的空间,将我们私有变量的值进行适当的处理即可。代码如下:

#include<iostream>
using namespace std;
//定义一个命名空间于,在其中创建一个字符串类
namespace test
{
    class string
    {
     private:
		int _size;
		int _capacity;
		char* _str;
     public:
        //使用函数重载,丰富构造函数,使我们可以使用多种类型进行对象的初始化
		string(const char ch)
		{
			_size = 1;
			_capacity = 1;
			_str = new char[2];
			_str[0] = ch;
			_str[1] = '\0';
		}
        //写一个构造函数,用于初始化string类对象
		string(const char*ch="")
		{
			_size = strlen(ch);
			_capacity = _size;
			_str = new char[_capacity+1];
			memcpy(_str, ch, _size+1);
		}
        //定义一个拷贝构造
		string(const string& s)
		{
			_size = s._size;
			_capacity = s._capacity;
			_str = new char[s._capacity + 1];
			memcpy(_str, s._str, _size + 1);
		}
        //定义一个析构函数,用于使用完毕的空间的释放
		~string()
		{
			_size = 0;
			_capacity = 0;
			delete[] _str;
			_str = nullptr;
		}
    };
}

    之后为了检验我们的代码是否可以正常的使用,所以我们还可以提前完善出 c_str() 的功能, c_str()函数的作用就是返回我们堆区开辟的字符串的指针,让我们的cout不用重载也可以打印出来,(因为指针是内置类型,cout可以打印出内置类型)。同样的,我们还可以设置出一个返回size的接口,和capacity的接口,仅仅返回size和capacity的值即可,代码如下:

#include<iostream>
using namespace std;
//定义一个命名空间于,在其中创建一个字符串类
namespace test
{
    class string
    {
     private:
		int _size;
		int _capacity;
		char* _str;
     public:
        //使用函数重载,丰富构造函数,使我们可以使用多种类型进行对象的初始化
		string(const char ch)
		{
			_size = 1;
			_capacity = 1;
			_str = new char[2];
			_str[0] = ch;
			_str[1] = '\0';
		}
        //写一个构造函数,用于初始化string类对象
		string(const char*ch="")
		{
			_size = strlen(ch);
			_capacity = _size;
			_str = new char[_capacity+1];
			memcpy(_str, ch, _size+1);
		}
        //定义一个拷贝构造
		string(const string& s)
		{
			_size = s._size;
			_capacity = s._capacity;
			_str = new char[s._capacity + 1];
			memcpy(_str, s._str, _size + 1);
		}
        //定义一个析构函数,用于使用完毕的空间的释放
		~string()
		{
			_size = 0;
			_capacity = 0;
			delete[] _str;
			_str = nullptr;
		}
        const char* c_str()const
		{
			return _str;
		}
        int size()const
		{
			return _size;
		}
		int capacity()const
		{
			return _capacity;
		}
    };
}

    之后我们就可以通过 c_str() 函数检测我们代码的正确性的,测试上述构造函数是否可以正常使用,测试结果如下:

  我们会发现代码运行一切正常。

  itertor迭代器

  在完成构造函数之后我们就可以完善我们的迭代器了。之前我们说过,迭代器的本质就是一个指针,我们可以使用一个指针进行模拟实现迭代器。使用typedef重命名char*为itertor迭代器,之后返回首指针和尾指针即可。所示代码如下:

#include<iostream>
using namespace std;
//定义一个命名空间于,在其中创建一个字符串类
namespace test
{
    class string
    {
     private:
		int _size;
		int _capacity;
		char* _str;
     public:
        typedef char* itertor;
		typedef const char* cosnt_itertor;
        //使用函数重载,丰富构造函数,使我们可以使用多种类型进行对象的初始化
		string(const char ch)
		{
			_size = 1;
			_capacity = 1;
			_str = new char[2];
			_str[0] = ch;
			_str[1] = '\0';
		}
        //写一个构造函数,用于初始化string类对象
		string(const char*ch="")
		{
			_size = strlen(ch);
			_capacity = _size;
			_str = new char[_capacity+1];
			memcpy(_str, ch, _size+1);
		}
        //定义一个拷贝构造
		string(const string& s)
		{
			_size = s._size;
			_capacity = s._capacity;
			_str = new char[s._capacity + 1];
			memcpy(_str, s._str, _size + 1);
		}
        //定义一个析构函数,用于使用完毕的空间的释放
		~string()
		{
			_size = 0;
			_capacity = 0;
			delete[] _str;
			_str = nullptr;
		}
        const char* c_str()const
		{
			return _str;
		}
        int size()const
		{
			return _size;
		}
		int capacity()const
		{
			return _capacity;
		}
        itertor begin()
		{
			return _str;
		}
		itertor end()
		{
			return (_str + _size);
		}
        const itertor begin() const 
		{
			return _str;
		}
		const itertor end()const
		{
			return (_str + _size);
		}
    };
}

  因为我们迭代器的作用是遍历我们的字符串数组,对于字符串数组我们可能进行的操作分为查找和修改两种,因此我们的迭代器也应该分为两种,一种是不使用const修饰的迭代器,另一种是使用const修饰的迭代器。使用const修饰的迭代器只能读取数据不能进行数据的修改。

  []操作符重载

  作为打印字符串指定元素并修改的操作,相信肯定离不开 [ ] 所以下一步我们要进行的操作就是重载 [ ] 运算符,进行指定元素的返回即可。这一部分的功能实现起来同样很简单,代码如下:

#include<iostream>
using namespace std;
//定义一个命名空间于,在其中创建一个字符串类
namespace test
{
    class string
    {
     private:
		int _size;
		int _capacity;
		char* _str;
     public:
        typedef char* itertor;
		typedef const char* cosnt_itertor;
        //使用函数重载,丰富构造函数,使我们可以使用多种类型进行对象的初始化
		string(const char ch)
		{
			_size = 1;
			_capacity = 1;
			_str = new char[2];
			_str[0] = ch;
			_str[1] = '\0';
		}
        //写一个构造函数,用于初始化string类对象
		string(const char*ch="")
		{
			_size = strlen(ch);
			_capacity = _size;
			_str = new char[_capacity+1];
			memcpy(_str, ch, _size+1);
		}
        //定义一个拷贝构造
		string(const string& s)
		{
			_size = s._size;
			_capacity = s._capacity;
			_str = new char[s._capacity + 1];
			memcpy(_str, s._str, _size + 1);
		}
        //定义一个析构函数,用于使用完毕的空间的释放
		~string()
		{
			_size = 0;
			_capacity = 0;
			delete[] _str;
			_str = nullptr;
		}
        const char* c_str()const
		{
			return _str;
		}
        int size()const
		{
			return _size;
		}
		int capacity()const
		{
			return _capacity;
		}
        itertor begin()
		{
			return _str;
		}
		itertor end()
		{
			return (_str + _size);
		}
        const itertor begin() const 
		{
			return _str;
		}
		const itertor end()const
		{
			return (_str + _size);
		}
        char& operator[](size_t index)
		{
			//检查开始的下标是否合法
			assert(index < _size);
			return _str[index];
		}

		const char& operator[](size_t index)const
		{
			//检查开始的下标是否合法
			assert(index < _size);
			return _str[index];
		}
    };
}

  在返回指定元素值的时候,我们最好先进行合法性检查,保证不会因为数组越界造成系统报错。同样的我们返回的内容也包括读写两种功能,因此我们将其分为两个函数完成。代码运行的结果:

  功能一切正常。 

  字符串修改函数

   1.尾插函数

  尾插函数的作用就是向字符串当中插入一个字符的数据。对于这个函数我们第一步需要进行的操作就是检查容量是否已满,如果申请的容量满的话就需要重新申请空间,之后将数据拷贝到新申请的空间在释放原有的空间,最后在插入我们想要插入的数据即可。代码如下:

#include<iostream>
using namespace std;
//定义一个命名空间于,在其中创建一个字符串类
namespace test
{
    class string
    {
     private:
		int _size;
		int _capacity;
		char* _str;
     public:
        typedef char* itertor;
		typedef const char* cosnt_itertor;
        //使用函数重载,丰富构造函数,使我们可以使用多种类型进行对象的初始化
		string(const char ch)
		{
			_size = 1;
			_capacity = 1;
			_str = new char[2];
			_str[0] = ch;
			_str[1] = '\0';
		}
        //写一个构造函数,用于初始化string类对象
		string(const char*ch="")
		{
			_size = strlen(ch);
			_capacity = _size;
			_str = new char[_capacity+1];
			memcpy(_str, ch, _size+1);
		}
        //定义一个拷贝构造
		string(const string& s)
		{
			_size = s._size;
			_capacity = s._capacity;
			_str = new char[s._capacity + 1];
			memcpy(_str, s._str, _size + 1);
		}
        //定义一个析构函数,用于使用完毕的空间的释放
		~string()
		{
			_size = 0;
			_capacity = 0;
			delete[] _str;
			_str = nullptr;
		}
        const char* c_str()const
		{
			return _str;
		}
        int size()const
		{
			return _size;
		}
		int capacity()const
		{
			return _capacity;
		}
        itertor begin()
		{
			return _str;
		}
		itertor end()
		{
			return (_str + _size);
		}
        const itertor begin() const 
		{
			return _str;
		}
		const itertor end()const
		{
			return (_str + _size);
		}
        char& operator[](size_t index)
		{
			//检查开始的下标是否合法
			assert(index < _size);
			return _str[index];
		}

		const char& operator[](size_t index)const
		{
			//检查开始的下标是否合法
			assert(index < _size);
			return _str[index];
		}
        //实现尾插函数
		void push_back(char c)
		{
			//检验容量是否已满
			if (_capacity == _size)
			{
				//开辟新的空间,每次根据原有空间的二倍进行开辟,并将原有数据转移至新的空间当中
				//所开辟的空间需要考虑到'\0'
				char* tmp = new char[_capacity == 0 ? 2 : _capacity * 2 + 1];
				memcpy(tmp, _str, _size);
				delete[] _str;
				_str = tmp;
				_capacity == 0 ? _capacity=1 : _capacity *= 2;  //扩容完毕
			}
			//开始将数据插入到字符串数组当中
			_str[_size] = c;
			_size++;
			_str[_size] = '\0';
		}
    };
}

  尝试插入一个元素测试上述代码的运行结果:

   代码运行依旧正常。

   2.append函数

  既然有一个字符的插入就肯定有一个字符串的插入,使用append函数就可以向对象当中插入一个字符串。我们同样需要先检查容量是否已满,如果容量已经满的话就需要重新开辟一个数组,重复我们上面所进行的操作即可。代码如下:
 

#include<iostream>
using namespace std;
//定义一个命名空间于,在其中创建一个字符串类
namespace test
{
    class string
    {
     private:
		int _size;
		int _capacity;
		char* _str;
     public:
        typedef char* itertor;
		typedef const char* cosnt_itertor;
        //使用函数重载,丰富构造函数,使我们可以使用多种类型进行对象的初始化
		string(const char ch)
		{
			_size = 1;
			_capacity = 1;
			_str = new char[2];
			_str[0] = ch;
			_str[1] = '\0';
		}
        //写一个构造函数,用于初始化string类对象
		string(const char*ch="")
		{
			_size = strlen(ch);
			_capacity = _size;
			_str = new char[_capacity+1];
			memcpy(_str, ch, _size+1);
		}
        //定义一个拷贝构造
		string(const string& s)
		{
			_size = s._size;
			_capacity = s._capacity;
			_str = new char[s._capacity + 1];
			memcpy(_str, s._str, _size + 1);
		}
        //定义一个析构函数,用于使用完毕的空间的释放
		~string()
		{
			_size = 0;
			_capacity = 0;
			delete[] _str;
			_str = nullptr;
		}
        const char* c_str()const
		{
			return _str;
		}
        int size()const
		{
			return _size;
		}
		int capacity()const
		{
			return _capacity;
		}
        itertor begin()
		{
			return _str;
		}
		itertor end()
		{
			return (_str + _size);
		}
        const itertor begin() const 
		{
			return _str;
		}
		const itertor end()const
		{
			return (_str + _size);
		}
        char& operator[](size_t index)
		{
			//检查开始的下标是否合法
			assert(index < _size);
			return _str[index];
		}

		const char& operator[](size_t index)const
		{
			//检查开始的下标是否合法
			assert(index < _size);
			return _str[index];
		}
        //实现尾插函数
		void push_back(char c)
		{
			//检验容量是否已满
			if (_capacity == _size)
			{
				//开辟新的空间,每次根据原有空间的二倍进行开辟,并将原有数据转移至新的空间当中
				//所开辟的空间需要考虑到'\0'
				char* tmp = new char[_capacity == 0 ? 2 : _capacity * 2 + 1];
				memcpy(tmp, _str, _size);
				delete[] _str;
				_str = tmp;
				_capacity == 0 ? _capacity=1 : _capacity *= 2;  //扩容完毕
			}
			//开始将数据插入到字符串数组当中
			_str[_size] = c;
			_size++;
			_str[_size] = '\0';
		}
        void append(const char* str)
		{
			//实现尾插一个新的字符串
			//检查数组的容量
			int len=strlen(str);
			if (_size + len > _capacity)
			{
				//开辟一段新的空间,转移数据后释放原有的空间
				char* tmp = new char[_size+len+1];
				memcpy(tmp, _str, _size);
				delete[] _str;
				_str = tmp;
				_capacity = _size + len;
			}
			//将字符串当中的数据拷贝到对象当中
			memcpy(&_str[_size], str, len);
			_size += len;
			_str[_size] = '\0';
		}
    };
}

  我们尝试向字符串当中插入一个字符串,得到的结果如下:

   代码运行的结果依旧正常。 

   3.+=运算符重载函数

  但是在string类当中最常用的尾插操作并不是append函数,因为每一次都需要调用函数书写上会很麻烦,因此我们重载了一个+=操作符。但是实质上我们在函数当中调用的还是append函数,只不过使用起来会简单很多。代码如下:

#include<iostream>
using namespace std;
//定义一个命名空间于,在其中创建一个字符串类
namespace test
{
    class string
    {
     private:
		int _size;
		int _capacity;
		char* _str;
     public:
        typedef char* itertor;
		typedef const char* cosnt_itertor;
        //使用函数重载,丰富构造函数,使我们可以使用多种类型进行对象的初始化
		string(const char ch)
		{
			_size = 1;
			_capacity = 1;
			_str = new char[2];
			_str[0] = ch;
			_str[1] = '\0';
		}
        //写一个构造函数,用于初始化string类对象
		string(const char*ch="")
		{
			_size = strlen(ch);
			_capacity = _size;
			_str = new char[_capacity+1];
			memcpy(_str, ch, _size+1);
		}
        //定义一个拷贝构造
		string(const string& s)
		{
			_size = s._size;
			_capacity = s._capacity;
			_str = new char[s._capacity + 1];
			memcpy(_str, s._str, _size + 1);
		}
        //定义一个析构函数,用于使用完毕的空间的释放
		~string()
		{
			_size = 0;
			_capacity = 0;
			delete[] _str;
			_str = nullptr;
		}
        const char* c_str()const
		{
			return _str;
		}
        int size()const
		{
			return _size;
		}
		int capacity()const
		{
			return _capacity;
		}
        itertor begin()
		{
			return _str;
		}
		itertor end()
		{
			return (_str + _size);
		}
        const itertor begin() const 
		{
			return _str;
		}
		const itertor end()const
		{
			return (_str + _size);
		}
        char& operator[](size_t index)
		{
			//检查开始的下标是否合法
			assert(index < _size);
			return _str[index];
		}

		const char& operator[](size_t index)const
		{
			//检查开始的下标是否合法
			assert(index < _size);
			return _str[index];
		}
        //实现尾插函数
		void push_back(char c)
		{
			//检验容量是否已满
			if (_capacity == _size)
			{
				//开辟新的空间,每次根据原有空间的二倍进行开辟,并将原有数据转移至新的空间当中
				//所开辟的空间需要考虑到'\0'
				char* tmp = new char[_capacity == 0 ? 2 : _capacity * 2 + 1];
				memcpy(tmp, _str, _size);
				delete[] _str;
				_str = tmp;
				_capacity == 0 ? _capacity=1 : _capacity *= 2;  //扩容完毕
			}
			//开始将数据插入到字符串数组当中
			_str[_size] = c;
			_size++;
			_str[_size] = '\0';
		}
        void append(const char* str)
		{
			//实现尾插一个新的字符串
			//检查数组的容量
			int len=strlen(str);
			if (_size + len > _capacity)
			{
				//开辟一段新的空间,转移数据后释放原有的空间
				char* tmp = new char[_size+len+1];
				memcpy(tmp, _str, _size);
				delete[] _str;
				_str = tmp;
				_capacity = _size + len;
			}
			//将字符串当中的数据拷贝到对象当中
			memcpy(&_str[_size], str, len);
			_size += len;
			_str[_size] = '\0';
		}
        string& operator+=(const char* str)
		{
			//复用append即可
			append(str);
			return *this;
		}
    };
}

  就像是我们上述代码当中的,我们只需要在函数当中复用append函数即可,尝试调用+=函数插入一个字符串,运行结果如下:

 代码运行结果一切正常。

   4.clear函数

  clear函数的作用就是清空一个对象当中所存储的所有数据,这一个操作同样很简单。我们只需要想清楚字符串存在的原理即可。字符串存在的标志有两个,第一 ‘ \0 ’, 第二size。想要将字符串当中的内容清空,我们只需要将size置为0,并将 str [0] 当中存入 ‘ \0 ’ 即可。代码如下:

#include<iostream>
using namespace std;
//定义一个命名空间于,在其中创建一个字符串类
namespace test
{
    class string
    {
     private:
		int _size;
		int _capacity;
		char* _str;
     public:
        typedef char* itertor;
		typedef const char* cosnt_itertor;
        //使用函数重载,丰富构造函数,使我们可以使用多种类型进行对象的初始化
		string(const char ch)
		{
			_size = 1;
			_capacity = 1;
			_str = new char[2];
			_str[0] = ch;
			_str[1] = '\0';
		}
        //写一个构造函数,用于初始化string类对象
		string(const char*ch="")
		{
			_size = strlen(ch);
			_capacity = _size;
			_str = new char[_capacity+1];
			memcpy(_str, ch, _size+1);
		}
        //定义一个拷贝构造
		string(const string& s)
		{
			_size = s._size;
			_capacity = s._capacity;
			_str = new char[s._capacity + 1];
			memcpy(_str, s._str, _size + 1);
		}
        //定义一个析构函数,用于使用完毕的空间的释放
		~string()
		{
			_size = 0;
			_capacity = 0;
			delete[] _str;
			_str = nullptr;
		}
        const char* c_str()const
		{
			return _str;
		}
        int size()const
		{
			return _size;
		}
		int capacity()const
		{
			return _capacity;
		}
        itertor begin()
		{
			return _str;
		}
		itertor end()
		{
			return (_str + _size);
		}
        const itertor begin() const 
		{
			return _str;
		}
		const itertor end()const
		{
			return (_str + _size);
		}
        char& operator[](size_t index)
		{
			//检查开始的下标是否合法
			assert(index < _size);
			return _str[index];
		}

		const char& operator[](size_t index)const
		{
			//检查开始的下标是否合法
			assert(index < _size);
			return _str[index];
		}
        //实现尾插函数
		void push_back(char c)
		{
			//检验容量是否已满
			if (_capacity == _size)
			{
				//开辟新的空间,每次根据原有空间的二倍进行开辟,并将原有数据转移至新的空间当中
				//所开辟的空间需要考虑到'\0'
				char* tmp = new char[_capacity == 0 ? 2 : _capacity * 2 + 1];
				memcpy(tmp, _str, _size);
				delete[] _str;
				_str = tmp;
				_capacity == 0 ? _capacity=1 : _capacity *= 2;  //扩容完毕
			}
			//开始将数据插入到字符串数组当中
			_str[_size] = c;
			_size++;
			_str[_size] = '\0';
		}
        void append(const char* str)
		{
			//实现尾插一个新的字符串
			//检查数组的容量
			int len=strlen(str);
			if (_size + len > _capacity)
			{
				//开辟一段新的空间,转移数据后释放原有的空间
				char* tmp = new char[_size+len+1];
				memcpy(tmp, _str, _size);
				delete[] _str;
				_str = tmp;
				_capacity = _size + len;
			}
			//将字符串当中的数据拷贝到对象当中
			memcpy(&_str[_size], str, len);
			_size += len;
			_str[_size] = '\0';
		}
        string& operator+=(const char* str)
		{
			//复用append即可
			append(str);
			return *this;
		}
        void clear()
		{
			//清空字符串对象当中的数据
			_str[0] = '\0';
			_size = 0;
		}
    };
}

  同样测试上述代码运行的效果如下:

  没有任何输出就是我们最好的结果。 

   5.swap函数

  之后是我们的交换函数。想要交换两个字符串对象当中存储的数据,我们实质上仅仅需要的只是交换两个对象当中存储的size和capacity以及指向特定空间的指针而已。至于交换这些内置类型的函数我们也不需要自己写,直接调用std命名空间域当中的swap函数使用即可。代码如下:
 

#include<iostream>
using namespace std;
//定义一个命名空间于,在其中创建一个字符串类
namespace test
{
    class string
    {
     private:
		int _size;
		int _capacity;
		char* _str;
     public:
        typedef char* itertor;
		typedef const char* cosnt_itertor;
        //使用函数重载,丰富构造函数,使我们可以使用多种类型进行对象的初始化
		string(const char ch)
		{
			_size = 1;
			_capacity = 1;
			_str = new char[2];
			_str[0] = ch;
			_str[1] = '\0';
		}
        //写一个构造函数,用于初始化string类对象
		string(const char*ch="")
		{
			_size = strlen(ch);
			_capacity = _size;
			_str = new char[_capacity+1];
			memcpy(_str, ch, _size+1);
		}
        //定义一个拷贝构造
		string(const string& s)
		{
			_size = s._size;
			_capacity = s._capacity;
			_str = new char[s._capacity + 1];
			memcpy(_str, s._str, _size + 1);
		}
        //定义一个析构函数,用于使用完毕的空间的释放
		~string()
		{
			_size = 0;
			_capacity = 0;
			delete[] _str;
			_str = nullptr;
		}
        const char* c_str()const
		{
			return _str;
		}
        int size()const
		{
			return _size;
		}
		int capacity()const
		{
			return _capacity;
		}
        itertor begin()
		{
			return _str;
		}
		itertor end()
		{
			return (_str + _size);
		}
        const itertor begin() const 
		{
			return _str;
		}
		const itertor end()const
		{
			return (_str + _size);
		}
        char& operator[](size_t index)
		{
			//检查开始的下标是否合法
			assert(index < _size);
			return _str[index];
		}

		const char& operator[](size_t index)const
		{
			//检查开始的下标是否合法
			assert(index < _size);
			return _str[index];
		}
        //实现尾插函数
		void push_back(char c)
		{
			//检验容量是否已满
			if (_capacity == _size)
			{
				//开辟新的空间,每次根据原有空间的二倍进行开辟,并将原有数据转移至新的空间当中
				//所开辟的空间需要考虑到'\0'
				char* tmp = new char[_capacity == 0 ? 2 : _capacity * 2 + 1];
				memcpy(tmp, _str, _size);
				delete[] _str;
				_str = tmp;
				_capacity == 0 ? _capacity=1 : _capacity *= 2;  //扩容完毕
			}
			//开始将数据插入到字符串数组当中
			_str[_size] = c;
			_size++;
			_str[_size] = '\0';
		}
        void append(const char* str)
		{
			//实现尾插一个新的字符串
			//检查数组的容量
			int len=strlen(str);
			if (_size + len > _capacity)
			{
				//开辟一段新的空间,转移数据后释放原有的空间
				char* tmp = new char[_size+len+1];
				memcpy(tmp, _str, _size);
				delete[] _str;
				_str = tmp;
				_capacity = _size + len;
			}
			//将字符串当中的数据拷贝到对象当中
			memcpy(&_str[_size], str, len);
			_size += len;
			_str[_size] = '\0';
		}
        string& operator+=(const char* str)
		{
			//复用append即可
			append(str);
			return *this;
		}
        void clear()
		{
			//清空字符串对象当中的数据
			_str[0] = '\0';
			_size = 0;
		}
        void swap(string& s)
		{
			//只需要交换指针指向并修改size和capacity即可,无需开辟新的空间
			std::swap(_size, s._size);
			std::swap(_capacity, s._capacity);
			std::swap(_str, s._str);
		}
    };
}

   测试上述代码运行的结果:

  字符串当中存储的数据也已经进行交换完毕了。代码运行一切正常。

  容量检测或修改函数

  在这一部分我们需要着重介绍的是resize和reserve函数。在之前我们向大家介绍过,resize函数的作用是调整字符串的长度,对字符串进行指定的修改。而reserve函数是针对容量进行修改,并不会对字符串造成影响。我们就来分别认识这两个函数。

  resize函数

  对于resize函数我们需要分情况进行讨论。当我们调整之后的大小小于我们原有的空间的时候,我们需要舍弃指定位置后面的字符串的内容。如果调整的大小大于我们原有的空间的时候,我们需要针对于容量进行适当的扩容操作。并且为了符合库当中的函数,我们还应该允许对增加的字符串插入指定的字符。只要分好类,代码的编写含是很简单的,代码如下:

#include<iostream>
using namespace std;
//定义一个命名空间于,在其中创建一个字符串类
namespace test
{
    class string
    {
     private:
		int _size;
		int _capacity;
		char* _str;
     public:
        typedef char* itertor;
		typedef const char* cosnt_itertor;
        //使用函数重载,丰富构造函数,使我们可以使用多种类型进行对象的初始化
		string(const char ch)
		{
			_size = 1;
			_capacity = 1;
			_str = new char[2];
			_str[0] = ch;
			_str[1] = '\0';
		}
        //写一个构造函数,用于初始化string类对象
		string(const char*ch="")
		{
			_size = strlen(ch);
			_capacity = _size;
			_str = new char[_capacity+1];
			memcpy(_str, ch, _size+1);
		}
        //定义一个拷贝构造
		string(const string& s)
		{
			_size = s._size;
			_capacity = s._capacity;
			_str = new char[s._capacity + 1];
			memcpy(_str, s._str, _size + 1);
		}
        //定义一个析构函数,用于使用完毕的空间的释放
		~string()
		{
			_size = 0;
			_capacity = 0;
			delete[] _str;
			_str = nullptr;
		}
        const char* c_str()const
		{
			return _str;
		}
        int size()const
		{
			return _size;
		}
		int capacity()const
		{
			return _capacity;
		}
        itertor begin()
		{
			return _str;
		}
		itertor end()
		{
			return (_str + _size);
		}
        const itertor begin() const 
		{
			return _str;
		}
		const itertor end()const
		{
			return (_str + _size);
		}
        char& operator[](size_t index)
		{
			//检查开始的下标是否合法
			assert(index < _size);
			return _str[index];
		}

		const char& operator[](size_t index)const
		{
			//检查开始的下标是否合法
			assert(index < _size);
			return _str[index];
		}
        //实现尾插函数
		void push_back(char c)
		{
			//检验容量是否已满
			if (_capacity == _size)
			{
				//开辟新的空间,每次根据原有空间的二倍进行开辟,并将原有数据转移至新的空间当中
				//所开辟的空间需要考虑到'\0'
				char* tmp = new char[_capacity == 0 ? 2 : _capacity * 2 + 1];
				memcpy(tmp, _str, _size);
				delete[] _str;
				_str = tmp;
				_capacity == 0 ? _capacity=1 : _capacity *= 2;  //扩容完毕
			}
			//开始将数据插入到字符串数组当中
			_str[_size] = c;
			_size++;
			_str[_size] = '\0';
		}
        void append(const char* str)
		{
			//实现尾插一个新的字符串
			//检查数组的容量
			int len=strlen(str);
			if (_size + len > _capacity)
			{
				//开辟一段新的空间,转移数据后释放原有的空间
				char* tmp = new char[_size+len+1];
				memcpy(tmp, _str, _size);
				delete[] _str;
				_str = tmp;
				_capacity = _size + len;
			}
			//将字符串当中的数据拷贝到对象当中
			memcpy(&_str[_size], str, len);
			_size += len;
			_str[_size] = '\0';
		}
        string& operator+=(const char* str)
		{
			//复用append即可
			append(str);
			return *this;
		}
        void clear()
		{
			//清空字符串对象当中的数据
			_str[0] = '\0';
			_size = 0;
		}
        void swap(string& s)
		{
			//只需要交换指针指向并修改size和capacity即可,无需开辟新的空间
			std::swap(_size, s._size);
			std::swap(_capacity, s._capacity);
			std::swap(_str, s._str);
		}
        void resize(size_t n, char c = '\0')
		{
			//当n小于原有字符的个数的时候,舍弃n后面的字符
			if (n < _size)
			{
				_size = n;
				_str[n] = '\0';
			}
			//当n大于原有字符的个数的时候,判断是否容量需要进行改变,适当进行扩容
			if (n > _size)
			{
				if (n > _capacity)
				{
					//需要进行扩容,直接调用reserve函数
					reserve(n);
				}
				//将默认字符填入申请的空间当中
				while (_size < n)
				{
					_str[_size] = c;
					_size++;
				}
				_str[_size] = '\0';
			}
		}
    };
}

运行结果如下:
    空间缩小就舍弃原有字符串当中存储的数据的内容,空间扩大允许插入指定的字符。带么运行一切正常。

  reserve函数

  reserve函数主要针对于改变字符串对象的容量。实质上就是调用new函数重新申请一个空间,之后将我们的数据全部转移到新开辟的数组当中即可,同样分为两种条件。第一种就是对字符串扩容,那么我们原本字符串当中的数据就不做改变。另一种就是缩小容量,并且缩小之后的容量小于我们的字符串的长度,这一种情况编译器当中并没有明确的规定,所以我们可以修改原本的字符串的打印顺序也可以不做修改。(最好进行修改,因为如果不修改,就会造成数组越界的问题。)实现这一部分功能的代码如下:
 

#include<iostream>
using namespace std;
//定义一个命名空间于,在其中创建一个字符串类
namespace test
{
    class string
    {
     private:
		int _size;
		int _capacity;
		char* _str;
     public:
        typedef char* itertor;
		typedef const char* cosnt_itertor;
        //使用函数重载,丰富构造函数,使我们可以使用多种类型进行对象的初始化
		string(const char ch)
		{
			_size = 1;
			_capacity = 1;
			_str = new char[2];
			_str[0] = ch;
			_str[1] = '\0';
		}
        //写一个构造函数,用于初始化string类对象
		string(const char*ch="")
		{
			_size = strlen(ch);
			_capacity = _size;
			_str = new char[_capacity+1];
			memcpy(_str, ch, _size+1);
		}
        //定义一个拷贝构造
		string(const string& s)
		{
			_size = s._size;
			_capacity = s._capacity;
			_str = new char[s._capacity + 1];
			memcpy(_str, s._str, _size + 1);
		}
        //定义一个析构函数,用于使用完毕的空间的释放
		~string()
		{
			_size = 0;
			_capacity = 0;
			delete[] _str;
			_str = nullptr;
		}
        const char* c_str()const
		{
			return _str;
		}
        int size()const
		{
			return _size;
		}
		int capacity()const
		{
			return _capacity;
		}
        itertor begin()
		{
			return _str;
		}
		itertor end()
		{
			return (_str + _size);
		}
        const itertor begin() const 
		{
			return _str;
		}
		const itertor end()const
		{
			return (_str + _size);
		}
        char& operator[](size_t index)
		{
			//检查开始的下标是否合法
			assert(index < _size);
			return _str[index];
		}

		const char& operator[](size_t index)const
		{
			//检查开始的下标是否合法
			assert(index < _size);
			return _str[index];
		}
        //实现尾插函数
		void push_back(char c)
		{
			//检验容量是否已满
			if (_capacity == _size)
			{
				//开辟新的空间,每次根据原有空间的二倍进行开辟,并将原有数据转移至新的空间当中
				//所开辟的空间需要考虑到'\0'
				char* tmp = new char[_capacity == 0 ? 2 : _capacity * 2 + 1];
				memcpy(tmp, _str, _size);
				delete[] _str;
				_str = tmp;
				_capacity == 0 ? _capacity=1 : _capacity *= 2;  //扩容完毕
			}
			//开始将数据插入到字符串数组当中
			_str[_size] = c;
			_size++;
			_str[_size] = '\0';
		}
        void append(const char* str)
		{
			//实现尾插一个新的字符串
			//检查数组的容量
			int len=strlen(str);
			if (_size + len > _capacity)
			{
				//开辟一段新的空间,转移数据后释放原有的空间
				char* tmp = new char[_size+len+1];
				memcpy(tmp, _str, _size);
				delete[] _str;
				_str = tmp;
				_capacity = _size + len;
			}
			//将字符串当中的数据拷贝到对象当中
			memcpy(&_str[_size], str, len);
			_size += len;
			_str[_size] = '\0';
		}
        string& operator+=(const char* str)
		{
			//复用append即可
			append(str);
			return *this;
		}
        void clear()
		{
			//清空字符串对象当中的数据
			_str[0] = '\0';
			_size = 0;
		}
        void swap(string& s)
		{
			//只需要交换指针指向并修改size和capacity即可,无需开辟新的空间
			std::swap(_size, s._size);
			std::swap(_capacity, s._capacity);
			std::swap(_str, s._str);
		}
        void resize(size_t n, char c = '\0')
		{
			//当n小于原有字符的个数的时候,舍弃n后面的字符
			if (n < _size)
			{
				_size = n;
				_str[n] = '\0';
			}
			//当n大于原有字符的个数的时候,判断是否容量需要进行改变,适当进行扩容
			if (n > _size)
			{
				if (n > _capacity)
				{
					//需要进行扩容,直接调用reserve函数
					reserve(n);
				}
				//将默认字符填入申请的空间当中
				while (_size < n)
				{
					_str[_size] = c;
					_size++;
				}
				_str[_size] = '\0';
			}
		}
        void reserve(size_t n)
		{
			//调整capacity的大小,一般情况下,容量变小不做改变
			if (_capacity < n)
			{
				char* tmp = new char[n + 1];
				_capacity = n;
				memcpy(tmp, _str, _size + 1);
				delete[] _str;
				_str = tmp;
			}
		}
    };
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/883581.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【C++从0到王者】第二十一站:继承

文章目录 前言一、继承的概念及定义1. 继承的概念2.继承的格式3.继承关系与访问限定符 二、基类和派生类的赋值转换三、继承中的作用域四、派生类的默认成员函数五、继承与友元六、继承与静态成员 前言 继承是面向对象的三大特性之一。我们常常会遇到这样的情况。很多角色的信…

显卡nvidia-smi后 提示Faild 解决过程,包含卸载重装NVIDIA驱动步骤

显卡异常: 显卡nvidia-smi后 提示Faild 解决过程&#xff0c;卸载重装nvidia驱动步骤 文章目录 显卡异常: 显卡nvidia-smi后 提示Faild 解决过程&#xff0c;卸载重装nvidia驱动步骤 [toc]1 缘由2 解决过程3 过程所需命令4 解决4.1 把该显卡重新拔插一下卸载NVIDIA驱动的方法&a…

远程遥控IPTables进行端口复用

一、配置&#xff08;通过ip进行ping&#xff09; 1.创建复用链 iptables -t nat -N LETMEIN 2.创建端口复用将流量转发到22端口上 iptables -t nat -A LETMEIN -p tcp -j REDIRECT --to-port 22 3.开启开关&#xff0c;如果接收到一个长为 1139 的 ICMP 包&#xff0c;则将…

Ajax及前端工程化

Ajax&#xff1a;异步的js与xml。 作用&#xff1a; 1、通过ajax给服务器发送数据&#xff0c;并获得其响应的数据。 2、可以在不更新整个网页的情况下&#xff0c;与服务器交换数据并更新部分网页的技术。 一、同步与异步 二、原生Ajax 1、准备数据地址 2、创建XMLHttpReq…

图神经网络 day2 图的分类

图神经网络基础算法 1 GCN2 GraphSAGE2.1 采样&#xff1a;采样固定长度的邻居2.2 聚合2.3 GraphSAGE_minibatch2.4 GraphSAGE_embedding 3 GAT4. 图网络的分类4.1 递归图神经网络 RGNN4.2 图卷积神经网络GCN4.3 图注意力网络 GAT4.4 图自动编码 GAE4.5 图时空网络 GSTN4.6 图生…

echarts 柱状图-折线图-饼图的基础使用

上图示例图表展示相关配置&#xff1a; var myChart echarts.init(this.$refs.firstMain);myChart.setOption({legend: { // 图例设置top: "15%",type: "scroll",orient: "vertical",//图例列表的布局朝向。left: "right",pageIconCo…

【Docker】 Docker-Composite 启动 WordPress

引 本文将使用流行的博客搭建工具 WordPress 搭建一个私人博客站点。部署过程中使用到了 Docker 、MySQL 。站点搭建完成后经行了发布文章的体验。 WordPress WordPress 是一个广泛使用的开源内容管理系统&#xff08;CMS&#xff09;&#xff0c;用于构建和管理网站、博客和…

ChatGPT​保密吗?它有哪些潜在风险?如何规避?

自2022年11月公开发布以来&#xff0c;ChatGPT已成为许多企业和个人的必备工具&#xff0c;但随着该技术越来越多地融入我们的日常生活&#xff0c;人们很自然地想知道&#xff1a;ChatGPT是否是保密的。 问&#xff1a;ChatGPT保密吗&#xff1f; 答&#xff1a;否&#xff0…

MIUI免费字体更换

一、打开主题壁纸 二、选择 热销字 三、点击右上角 搜索 四、输入 字体 可以看到&#xff0c;免费的字体没多少&#xff0c;此时这里可以输入其他关键词&#xff1a;拼音、手写等&#xff0c;看个人需求进行筛选免费即可 关键词有以下这些&#xff0c;但不局限这些哈 五、点击…

linux系统服务学习(六)FTP服务学习

文章目录 FTP、NFS、SAMBA系统服务一、FTP服务概述1、FTP服务介绍2、FTP服务的客户端工具3、FTP的两种运行模式&#xff08;了解&#xff09;☆ 主动模式☆ 被动模式 4、搭建FTP服务&#xff08;重要&#xff09;5、FTP的配置文件详解&#xff08;重要&#xff09; 二、FTP任务…

共读《科研论文配图绘制指南--基于Python》学习重点

Book 《科研论文配图绘制指南–基于Python》 特别提示 学习内容&#xff08;书籍前3章&#xff09;开营时在群内以PDF形式发放 课程背景 系统地介绍基于Python的科研论文配图的绘制技巧&#xff0c;提高科研工作者的绘图效率&#xff1b; 100多种图形的详细绘制方法&#…

STM32F103-OLED使用教程

目录 1. OLED屏介绍2. OLED如何显示一个点3. 配置OLED屏幕4. OLED显示字符串和汉字5. OLED屏幕显示图片6. 总结 1. OLED屏介绍 OLED&#xff08;Organic Light Emitting Diode&#xff09;&#xff1a;有机发光二极管OLED显示屏&#xff1a;性能优异的新型显示屏&#xff0c;具…

Vue组件(详解)

目录 组件&#xff1a; 全局组件&#xff1a; 在HTML页面声明template&#xff1a; 局部组件&#xff1a; 局部组件第一种方式&#xff1a; 局部组件第二种方式&#xff1a; 插槽slot&#xff1a; 匿名插槽&#xff1a; 具名插槽&#xff1a; 父子组件通信&#xff1…

【双指针_和为 s 的两个数_C++】

和为s的两个数字 class Solution { public:vector<int> twoSum(vector<int>& nums, int target) {int n nums.size();int left 0;int right n-1;while(left<right){if(nums[left]nums[right]>target) right--;else if(nums[left]nums[right]<tar…

Postman接口自动化测试实例

一.实例背景 在实际业务中&#xff0c;经常会出现让用户输入用户密码进行验证的场景。而为了安全&#xff0c;一般都会先请求后台服务器获取一个随机数做为盐值&#xff0c;然后将盐值和用户输入的密码通过前端的加密算法生成加密后串传给后台服务器&#xff0c;后台服务器接到…

车载以太网物理层

车载以太网物理层 O S I 参考模型的第 1 层&#xff08; 最底层&#xff09;。负责逻辑信号&#xff08; 比特流&#xff09;与物理信号&#xff08;电信号、光信号&#xff09;之间的互相转换&#xff0c;通过传输介质为数据链路层提供物理连接。 车载以太网与传统以太网相比…

matlab使用教程(16)—图论中图的定义与修改

1.修改现有图的节点和边 此示例演示如何使用 addedge 、 rmedge 、 addnode 、 rmnode 、 findedge 、 findnode 及 subgraph 函数访问和修改 graph 或 digraph 对象中的节点和/或边。 1.1 添加节点 创建一个包含四个节点和四条边的图。s 和 t 中的对应元素用于指定每条…

【教程】零成本将小米净化器改造为无叶风扇

某宝某多上&#xff0c;就这么点破塑料&#xff0c;就要买79&#xff1f;&#xff01;&#xff01; 我这枚韭菜可不上当。咱自己做一个&#xff01; 真香~

BBS项目day02、注册、登录(登录之随机验证码)、修改密码、退出登录、密码加密加盐

一、注册 1.注册之前端页面 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><title>注册页面</title><!--动态引入文件-->{% load static %}<script src"{% static js/jquery.min.js %…

jquery技术学习2

移动节点 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><title>移动节点</title><script type"text/javascript" src"../script/jquery-3.6.0.min.js"></script>&l…