【CPP】string 类的模拟实现(上)

简介: 【CPP】string 类的模拟实现(上)

👉前言👈


因为 string 类的函数接口组合在一起才好玩,所以我们模拟实现 string 类就一组一组函数来模拟实现。为了避免我们模拟实现的 string 类和库里的 string 类冲突,所以我们将自己实现的 string 类封装在命名空间里。


👉访问和遍历 string 类👈


首先,string 类是一个可以动态增长的字符数组,其实就相当于是存储字符的顺序表。那么 string 类的成员变量如下方代码:


namespace Joy
{
  class string
  {
  public:
    // 成员函数
  private:
    char* _str; // 指向动态开辟的数组
    size_t _size; // 数组的有效字符个数
    size_t _capacity; // 数组的容量
  };
}


那现在我们就来模拟实现 string 类了。string 类最重要的就是构造函数了,所以我们先实现构造函数。


string 类的构造函数


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


注:string类的构造函数最好提供空串缺省参数,然后还有多开一个空间来存储\0标识字符,标识字符不算是 string 类的有效字符,给 _size 和 _capacity 都赋好初值后,就借助 strcpy 函数来拷贝字符串了。


string 类的析构函数


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


为了 string 类能用起来像数组一样,我们需要提供以下函数接口size、c_str和[]运算符重载。


const char* c_str() const
{
  return _str; // 返回数组首元素的地址
}
size_t size() const
{
  return _size;
}
// 普通对象:可读可写
char& operator[](size_t pos)
{
  assert(pos < _size);
  return _str[pos];
}
// const对象:只读
const char& operator[](size_t pos) const
{
  assert(pos < _size);
  return _str[pos];
}


以上实现的函数接口,足以帮我构造一个 string 类对象和访问 string 类对象了,那么现在我们就来测试一下。


22b031085b3649caa819ad1f76da2612.png


这样,访问和遍历 string 类对象的第一种方式就搞定了。那我们现在来实现一下遍历 string 类对象的第二种方式:迭代器iterator


迭代器可能是指针,也有可能不是指针,所以我们实现 string 类迭代器的方式是指针。


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;
}


那现在我们来测试一下功能。


e4fff4e76ef24147bd4488c7b2e17e0a.png


注:以上的 string 类迭代器是使用指针实现的,但是库里的 string 类的迭代器就有可能不是用指针实现了,比如 string 类的反向迭代器就不是使用指针来实现的了。因为reverse_iterator rit = s.rbegin() rit++会走到s.rend(),而如果使用指针实现,应该it--才能走到s.end(),使用 string 类的反向迭代器不是使用指针来实现的。


现在我们已经实现了 string 类的正向迭代器,其实范围 for 也实现好了。因为范围 for 的底层原理就是迭代器,编译器会自动将范围 for 替换成迭代器。

f3a36862365c4ef58700231b56c3481b.png


如果我们将正向迭代器的代码屏蔽掉,那范围 for 的代码就无法编译通过了。


👉capacity、resize 和 reserve👈


size_t capacity() const
{
  return _capacity;
}
void reserve(size_t n)
{
  if (n > _capacity)
  {
    char* tmp = new char[n + 1];
    strcpy(tmp, _str);
    delete[] _str;
    _str = tmp;
    _capacity = n;
  }
}
void resize(size_t n, char ch = '\0')
{
  if (n > _size)
  {
    reserve(n);
    for (size_t i = _size; i < n; i++)
    {
      _str[i] = ch;
    }
    _size = n;
    _str[_size] = '\0';
  }
  else
  {
    _str[n] = '\0';
    _size = n;
  }
}


reserve 函数接口说明


如果 n > _capacity,那么就先申请一块 n + 1的空间tmp,然后将数据拷贝到这块空间,_str = tmp,更新容量的大小_capacity = n。

如果n <= _capacity,那么就什么都不用做。如果缩容的话,就会造成效率低下。


resize 函数接口说明


  • n > _size时,调用 reserve 函数调整容量的大小,让插入字符 ch。
  • n <= _size时,直接_str[_size] = '\0'_size = n,注意:这个过程也不要缩容。

900c060e16b94b21ba147d198f02b716.png


👉string 类插入字符或字符串👈


我们主要模拟实现最常用的几个插入的接口,目的不是造更好的轮子。那我们现在开干!


string& push_back(char ch)
{
  if (_size == _capacity)
  {
    int newCapacity = _capacity == 0 ? 4 : _capacity * 2;
    reserve(newCapacity);
  }
  _str[_size] = ch;
  ++_size;
  _str[_size] = '\0';
  return *this;
}
string& append(const char* str)
{
  size_t len = strlen(str);
  if (_size + len > _capacity)
  {
    reserve(_size + len);
  }
  strcpy(_str + _size, str);
  _size += len;
  return *this;
}
string& operator+=(char ch)
{
  push_back(ch);
  return *this;
}
string& operator+=(const char* str)
{
  append(str);
  return *this;
}


注:不管是 push_back 函数,还是 append 函数,都要考虑是否需要扩容(reverse 函数会多开一个空间存储标识字符'\0'),然后再插入数据,最后更新_size和加上标识字符'\0'。对于operator +=运算符重载就可以赋用 push_back 函数和 append 函数了。

8c9d1eda58154075936376ce7b376d51.png


除了以上在尾部的插入,还有在任意位置的插入。


string& insert(size_t pos, char ch)
{
  assert(pos <= _size);
  if (_size == _capacity)
  {
    int newCapacity = _capacity == 0 ? 4 : _capacity * 2;
    reserve(newCapacity);
  }
  // 挪动数据
  /*
  int end = _size;
  while (end >= (int)pos)
  {
    _str[end + 1] = _str[end];
    --end;
  }
  _str[pos] = ch;
  ++_size;
  return *this;
  */
  size_t end = _size + 1;
  while (end > pos)
  {
    _str[end] = _str[end - 1];
    --end;
  }
  _str[pos] = ch;
  ++_size;
  return *this;
}
string& insert(size_t pos, const char* str)
{
  assert(pos <= _size);
  int len = strlen(str);
  if (_size + len > _capacity)
  {
    reserve(_size + len);
  }
  // 挪动数据
  /*
  int end = _size;
  while (end >= (int)pos)
  {
    _str[end + len] = _str[end];
    --end;
  }
  strncpy(_str + pos, str, len);
  _size += len;
  return *this;
  */
  size_t end = _size + len;
  while (end > pos + len - 1)
  {
    _str[end] = _str[end - len];
    --end;
  }
  strncpy(_str + pos, str, len);
  _size += len;
  return *this;
}


注:挪动数据就要注意对边界条件的控制,如果控制不会就会造成死循环和越界访问的问题。还有拷贝数据,不能使用strcpy函数,strcpy函数会将\0'拷贝过去。如果是>=,要讲pos强转为int。

288438c7c4754c6199a7dae234c99cae.png


👉erase👈


bool empty() const
{
  return _size == 0;
}
string& erase(size_t pos, size_t len = npos)
{
  assert(!empty());
  assert(pos < _size);
  if (len == npos || pos + len >= _size)
  {
    _str[pos] = '\0';
    _size = pos;
  }
  else
  {
    strcpy(_str + pos, _str + pos + len);
    _size -= len;
  }
  return *this;
}


注:const static size_t npos = -1npos是 string 类的静态成员变量,其值为-1。


erase 函数接口说明

当len == nps || pos + len >= _size时,就相当于将pos位置及其之后的字符都删掉,所以此时只需要将pos位置的字符改为'\0'并更新_size的值为pos。

除此之外,用 strcpy 函数将后面的字符向前覆盖删除,然后_size -= len就行了。

79aa90cc599e4e28a8bd1eff2cb3c4b7.png














相关文章
|
4月前
|
Java 索引
java基础(13)String类
本文介绍了Java中String类的多种操作方法,包括字符串拼接、获取长度、去除空格、替换、截取、分割、比较和查找字符等。
49 0
java基础(13)String类
|
2月前
|
存储 编译器 C语言
【c++丨STL】string类的使用
本文介绍了C++中`string`类的基本概念及其主要接口。`string`类在C++标准库中扮演着重要角色,它提供了比C语言中字符串处理函数更丰富、安全和便捷的功能。文章详细讲解了`string`类的构造函数、赋值运算符、容量管理接口、元素访问及遍历方法、字符串修改操作、字符串运算接口、常量成员和非成员函数等内容。通过实例演示了如何使用这些接口进行字符串的创建、修改、查找和比较等操作,帮助读者更好地理解和掌握`string`类的应用。
63 2
|
3月前
|
Java
【编程基础知识】(讲解+示例实战)方法参数的传递机制(值传递及地址传递)以及String类的对象的不可变性
本文深入探讨了Java中方法参数的传递机制,包括值传递和引用传递的区别,以及String类对象的不可变性。通过详细讲解和示例代码,帮助读者理解参数传递的内部原理,并掌握在实际编程中正确处理参数传递的方法。关键词:Java, 方法参数传递, 值传递, 引用传递, String不可变性。
74 1
【编程基础知识】(讲解+示例实战)方法参数的传递机制(值传递及地址传递)以及String类的对象的不可变性
|
3月前
|
安全 Java 测试技术
Java零基础-StringBuffer 类详解
【10月更文挑战第9天】Java零基础教学篇,手把手实践教学!
65 2
|
4月前
|
安全 Java
String类-知识回顾①
这篇文章回顾了Java中String类的相关知识点,包括`==`操作符和`equals()`方法的区别、String类对象的不可变性及其好处、String常量池的概念,以及String对象的加法操作。文章通过代码示例详细解释了这些概念,并探讨了使用String常量池时的一些行为。
String类-知识回顾①
|
3月前
|
存储 安全 C++
【C++打怪之路Lv8】-- string类
【C++打怪之路Lv8】-- string类
33 1
|
3月前
|
数据可视化 Java
让星星月亮告诉你,通过反射创建类的实例对象,并通过Unsafe theUnsafe来修改实例对象的私有的String类型的成员属性的值
本文介绍了如何使用 Unsafe 类通过反射机制修改对象的私有属性值。主要包括: 1. 获取 Unsafe 的 theUnsafe 属性:通过反射获取 Unsafe类的私有静态属性theUnsafe,并放开其访问权限,以便后续操作 2. 利用反射创建 User 类的实例对象:通过反射创建User类的实例对象,并定义预期值 3. 利用反射获取实例对象的name属性并修改:通过反射获取 User类实例对象的私有属性name,使用 Unsafe`的compareAndSwapObject方法直接在内存地址上修改属性值 核心代码展示了详细的步骤和逻辑,确保了对私有属性的修改不受 JVM 访问权限的限制
71 4
|
3月前
|
存储 安全 Java
【一步一步了解Java系列】:认识String类
【一步一步了解Java系列】:认识String类
35 2
|
3月前
|
安全 C语言 C++
【C++篇】探寻C++ STL之美:从string类的基础到高级操作的全面解析
【C++篇】探寻C++ STL之美:从string类的基础到高级操作的全面解析
57 4
|
4月前
|
存储 安全 Java
Java——String类详解
String 是 Java 中的一个类,用于表示字符串,属于引用数据类型。字符串可以通过多种方式定义,如直接赋值、创建对象、传入 char 或 byte 类型数组。直接赋值会将字符串存储在串池中,复用相同的字符串以节省内存。String 类提供了丰富的方法,如比较(equals() 和 compareTo())、查找(charAt() 和 indexOf())、转换(valueOf() 和 format())、拆分(split())和截取(substring())。此外,还介绍了 StringBuilder 和 StringJoiner 类,前者用于高效拼接字符串,后者用于按指定格式拼接字符串
167 1
Java——String类详解