string类的模拟实现(中)

简介: string类的模拟实现(中)

operator+=的实现

我们实现了push_back和append之后就可以直接复用这两个函数,实现operator+=

string& operator+=(char ch)
{
  push_back(ch);  //插入一个字符的时候
  return *this;
}
string& operator+=(const char* str)
{
  append(str); //插入字符串的时候
  return *this;
}//因为string对象在该函数之后不会释放空间,所以传引用返回,提高效率

string常用函数的实现

下面主要是对于insert、find、erase、substr、resize、一系列重载运算符等的实现

insert的实现

我们主要实现两种insert函数 1.在pos位置上插入n个字符c 2.在pos位置上插入字符串str

string& insert(size_t pos, size_t n, char c)
{
  //1.先判定pos是否正确
  assert(pos <= _size);
  //2.扩容
  reserve(_size + n);
  //3.在pos位置上开始挪动n个字符
  size_t end = _size;
  //因为如果pos为0的时候,无符号整型0减去1,end >= pos比较 为一个巨大值,使得该循环无法停止
  //npos是static变量,定义为-1;
  while (end >= pos && end != npos)
  {
    _str[end + n] = _str[end];
    --end;
  }
  //4.添加n个字符
  for (int i = 0; i < n; i++)
  {
    _str[pos + i] = c;
  }
  _size += n;
  return *this;
}
string& insert(size_t pos, const char*str)
{
  //1.pos的判定
  assert(pos <= _size);
  //2.扩容
  int len = strlen(str);
  reserve(_size + len);
  //3.在pos位置上移动len个字符
  size_t end = _size;
  while (end >= pos && end != npos)
  {
    _str[end + len] = _str[end];
    --end;
  }
  //4.将str字符串的字符依次输入
  for (int i = 0; i < len; i++)
  {
    _str[pos + i] = str[i];
  }
  //5.最后_size增加
  _size += len;
  return *this;
}

注意:npos的使用,是为了防止size_t无符号整型在于整型pos比较时候的强制转换,整型提升,得到一个巨大值,造成无限循环。

erase的实现

主要就是判断len是否等于npos,或者pos+len>=_size,分两种情况,是否从pos删除完,实际上就是在pos位置上加上'\0'即可,反之就是间隔len个距离向前移动字符,直到pos+len<=_size,最后_size-=len

string& erase(size_t pos = 0, size_t len = npos)
{
  assert(pos <= _size);
  if (len == npos || pos + len >= _size)
  {
    //表示从pos位置删完
    _str[pos] = '\0';
    _size = pos;
  }
  else {
    //从pos位置删除len个字符
    //向前挪动
    size_t end = pos;
    while (end+len <= _size)  
    {
      _str[end] = _str[end + len];
      end++;
    }
    //此时end==_size
    //_str[end] = '\0';
    _size -= len;
  }
  return *this;
}

find的实现

find的实现,就是遍历找到符合条件的下标,并返回

size_t find(char ch, size_t pos = 0) const
{
  assert(pos < _size);
  for (size_t i = pos; i < _size; i++)
  {
    if (_str[i] == ch)
    {
      return i;
    }
  }
  return npos;  //没有找到返回-1;
}
size_t find(const char* s, size_t pos = 0) const
{
  //使用strstr
  assert(pos < _size);
  const char* str = strstr(_str+pos, s);  //使用str函数,进行比较是否有对应的字符串
  if (str)
  {
    return str-_str;//两个指针相减,得到的是地址的偏移量
  }
  else {
    return npos;
  }
}

substr的实现

substr的实现,就是判断要解决的n的数值,然后新建一个string字符串,将从pos位置开始的n个字符依次添加到这个新字符串中,最后返回新字符串

//substr的实现
string substr(size_t pos = 0, size_t len = npos)
{
  assert(pos < _size);
  size_t n = len;
  //如果缺省len=npos 或者是截取的范围大于_size
  if (len == npos || pos + len >= _size)
  {
    n = _size - pos;  
  }
  //创建一个新的字符数组
  string new_str;
  new_str.reserve(n);
  for (size_t i = pos; i < n + pos; i++)
  {
    new_str += _str[i];
  }
  return new_str;
}

resize的实现

resize底层是有reserve的,即需要判断是否需要扩容,满足_size<=_capacity

//实现resize
void resize(size_t n, char ch = '\0')
{
  //两种情况,1.n<_size 直接赋值'\0' 2.判断是否扩容 
  if (n < _size)
  {
    _size = n;
    _str[_size] = '\0';
  }
  else
  {
    reserve(n);//让reserve来判断是否是需要扩容
    for (size_t i = _size; i < n; i++)
    {
      _str[i] = ch;
    }
    _size = n;
    _str[_size] = '\0';
  }
}

opeartor重载运算符

重载运算符,只要实现一两个就能实现其他,下面我们实现的是operator< 和operator== 然后通过调用这两个函数,来实现其他operator

//字符串比较按照ascii比较
    //bool operator<(const string& s)
    //{
    //  int num=memcmp(_str, s._str, _size > s._size?s._size : _size);
    //  //如果在最小长度下,前面数值小于后者 num返回的是负数
    //  
    //  return num == 0 ? _size < s._size : num < 0;
    //  //如果如果num为0,说明等于,且前者长度小于后者,返回真值,反之返回
    //}
bool operator<(const string& s)
{
  size_t i1 = 0;
  size_t i2 = 0;
  int num = _size > s._size ? s._size : _size;  //得到两者最小的长度
  while (i1 < num && i2 < num)
  {
    if (_str[i1] < s._str[i2])    //只要不相等就返回
    {
      return true;
    }
    else if(_str[i1] > s._str[i2]) {
      return false;
    }
    else
    {
      ++i1;   //该字符相当,那么继续++
      ++i2;
    }
  }
  return _size < s._size;  //现在退出循环,说明前num个字符都相等,如果此时_size<s._size 那么返回真,反之返回假
}
bool operator==(const string& s)
{
  return _size == s._size && memcmp(_str, s._str, _size > s._size ? s._size : _size) == 0;  //两者字符长度相等,且通过memcpy返回值是否为0来判断函数返回值
}
//我们把 _size == s._size 放在前面,那么后面只需要memcpy(_str,s._str,_size)==0即可
bool operator<=(const string& s)
{
  return *this < s || *this == s;
}
bool operator>(const string& s)
{
  return !(*this <= s);
}
bool operator>=(const string& s) 
{
  return !(*this < s);
}
//访问指定下标的字符
//operator[]的实现
// 
char& operator[](size_t pos)
{
  //可读写   pos表示下标
  assert(pos < _size);
  return _str[pos];  //返回的是单个字符所以用char  且_str变量离开该函数依旧存在,可以使用&返回
}
string& operator=(const string& s)
{
  if (this != &s)
  {
    //调用拷贝构造函数 将s的数据给tmp
    string tmp(s);
    std::swap(_str, tmp._str);
    std::swap(_size, tmp._size);
    std::swap(_capacity, tmp._capacity);
    //进行交换,交换之后tmp在函数结束之后就会释放空间,但是其通过拷贝构造函数生成的新的string对象中的数值留给了*this对象
  }
  return *this;
}

流插入>>和流提取<<

//流提取
ostream& operator<<(ostream& out, const String::string& s)
{
  //就是将s字符串中的每一个字符都加载到out中
  for (auto ch : s)
  {
    out << ch;
  }
  return out;
}
//流插入
istream& operator>>(istream& in, String::string& s)
{
  //判断一个字符是否结束 按照空格或者\0来判断
  s.clear();
  char buff [128];
  char ch = in.get();//get 字符
  int i = 0;
  while (ch == ' '|| ch == '\n')
  {
    ch = in.get();  //处理缓冲区前面的空格和换行
  }
  while (ch != ' ' && ch != '\n')
  {
    buff[i++] = ch;
  //如果输入的数值在127之外
    if (i == 127)  //先i++  相当于从1到127
    {
      //留出来一个空间给\0 所以只能这样
      buff[i] = '\0';
      s += buff;
      i = 0;//要重置i
    }
    ch = in.get();
  }
  if (i != 0)
  {
    //如果i不0的话,那就是说数值长度在127之内 直接扩容
    buff[i] = '\0';
    s += buff;
  }
  return in;
}

总结:

  • 有关于插入以及添加字符、字符串的函数,都需要考虑容量的问题所以底层会有reserve,比如push_back、append、resize、insert、+=等函数
  • find的实现就是遍历,以及使用strstr函数,快速方便的得到下标
  • erase、insert等函数都会涉及到移动数组元素,erase向前移动,insert向后移动
  • 重载运算符的实现,可以方便我们使用string类
相关文章
|
3月前
|
Java 索引
java基础(13)String类
本文介绍了Java中String类的多种操作方法,包括字符串拼接、获取长度、去除空格、替换、截取、分割、比较和查找字符等。
47 0
java基础(13)String类
|
28天前
|
存储 编译器 C语言
【c++丨STL】string类的使用
本文介绍了C++中`string`类的基本概念及其主要接口。`string`类在C++标准库中扮演着重要角色,它提供了比C语言中字符串处理函数更丰富、安全和便捷的功能。文章详细讲解了`string`类的构造函数、赋值运算符、容量管理接口、元素访问及遍历方法、字符串修改操作、字符串运算接口、常量成员和非成员函数等内容。通过实例演示了如何使用这些接口进行字符串的创建、修改、查找和比较等操作,帮助读者更好地理解和掌握`string`类的应用。
49 2
|
2月前
|
Java
【编程基础知识】(讲解+示例实战)方法参数的传递机制(值传递及地址传递)以及String类的对象的不可变性
本文深入探讨了Java中方法参数的传递机制,包括值传递和引用传递的区别,以及String类对象的不可变性。通过详细讲解和示例代码,帮助读者理解参数传递的内部原理,并掌握在实际编程中正确处理参数传递的方法。关键词:Java, 方法参数传递, 值传递, 引用传递, String不可变性。
68 1
【编程基础知识】(讲解+示例实战)方法参数的传递机制(值传递及地址传递)以及String类的对象的不可变性
|
2月前
|
安全 Java 测试技术
Java零基础-StringBuffer 类详解
【10月更文挑战第9天】Java零基础教学篇,手把手实践教学!
54 2
|
3月前
|
安全 Java
String类-知识回顾①
这篇文章回顾了Java中String类的相关知识点,包括`==`操作符和`equals()`方法的区别、String类对象的不可变性及其好处、String常量池的概念,以及String对象的加法操作。文章通过代码示例详细解释了这些概念,并探讨了使用String常量池时的一些行为。
String类-知识回顾①
|
2月前
|
存储 安全 C++
【C++打怪之路Lv8】-- string类
【C++打怪之路Lv8】-- string类
29 1
|
2月前
|
数据可视化 Java
让星星月亮告诉你,通过反射创建类的实例对象,并通过Unsafe theUnsafe来修改实例对象的私有的String类型的成员属性的值
本文介绍了如何使用 Unsafe 类通过反射机制修改对象的私有属性值。主要包括: 1. 获取 Unsafe 的 theUnsafe 属性:通过反射获取 Unsafe类的私有静态属性theUnsafe,并放开其访问权限,以便后续操作 2. 利用反射创建 User 类的实例对象:通过反射创建User类的实例对象,并定义预期值 3. 利用反射获取实例对象的name属性并修改:通过反射获取 User类实例对象的私有属性name,使用 Unsafe`的compareAndSwapObject方法直接在内存地址上修改属性值 核心代码展示了详细的步骤和逻辑,确保了对私有属性的修改不受 JVM 访问权限的限制
64 4
|
2月前
|
存储 安全 Java
【一步一步了解Java系列】:认识String类
【一步一步了解Java系列】:认识String类
31 2
|
2月前
|
安全 C语言 C++
【C++篇】探寻C++ STL之美:从string类的基础到高级操作的全面解析
【C++篇】探寻C++ STL之美:从string类的基础到高级操作的全面解析
54 4
|
2月前
|
存储 编译器 程序员
【C++篇】手撕 C++ string 类:从零实现到深入剖析的模拟之路
【C++篇】手撕 C++ string 类:从零实现到深入剖析的模拟之路
80 2