string类的模拟实现(下)

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

最后附上完整模拟实现string类的代码

#define _CRT_SECURE_NO_WARNINGS
#include<iostream>
#include<assert.h>
#include<math.h>
//模拟实现string
namespace String {
  class string {
  public:
    //迭代器: string中的迭代器实际上就是指针
    typedef char* iterator;
    typedef const char* const_iterator;
    iterator begin()
    {
      //begin 表示的是string的首元素地址
      return _str;
    }
    iterator end()
    {
      //end 返回string最后一个元素的下一个位置,也就是'\0'
      return _str + _size;
    }
    const_iterator begin() const
    {
      //begin 表示的是string的首元素地址
      return _str;
    }
    const_iterator end() const
    {
      //end 返回string最后一个元素的下一个位置,也就是'\0'
      return _str + _size;
    }
    //默认构造函数
    //string() 
    //  :_str(new char[1])
    //  ,size(0)
    //  ,capacity(0)
    //{
    //  _str[0] = '\0';
    //}
    //带参构造函数  字符串
    //string(const char* str)
    //  :_str(new char[strlen(str)])
    //  ,size(strlen(str))
    //  ,capacity(strlen(str))
    //{
    //  //存储字符串
    //  strcpy(_str, str);
    //}
    //默认构造和带参构造合并,使用缺省参数
    string(const char* str="")  //""字符串自带'\0'
      :_str(new char[strlen(str)+1])
      , _size(strlen(str))
      , _capacity(strlen(str))
    {
      //存储字符串
      strcpy(_str, str);
    }
    //拷贝构造
    //string(const string& s)
    //{
    //  //深拷贝,就是创建一个大小一样的空间
    //  _str = new char[s._capacity + 1];
    //  strcpy(_str, s._str);
    //  _size = s._size;
    //  _capacity = s._capacity;
    //}
    string(const string& s)
    {
      _str = new char[s._capacity + 1];
      memcpy(_str, s._str,s._size + 1);
      _size = s._size;
      _capacity = s._capacity;
    }
    //析构函数
    ~string()
    {
      delete[] _str;
      _str = nullptr;
      _size = _capacity = 0;
    }
    void Print() {
      cout << _str <<"\t" << _size <<"\t"<< _capacity << endl;
    }
    //size_t  无符号整型
    const char* c_str()
    {
      return _str;
    }
    //返回size
    size_t size() const  //const表示修饰this指针,也就是说只读,如果是const对象,也可以访问,普通用户相当于权限的缩小也可也访问
    {
      return _size;
    }
    //operator[]的实现
    // 
    char& operator[](size_t pos)
    {
      //可读写   pos表示下标
      assert(pos < _size);
      return _str[pos];  //返回的是单个字符所以用char  且_str变量离开该函数依旧存在,可以使用&返回
    }
    //对于const对象
    const char& operator[](size_t pos) const   //修饰const对象,前面const修饰的话,表示不可以被修改,后面const对象就修饰this指针  const String::string* this
    {
      //只读
      assert(pos < _size);
      return _str[pos];
    }
    //reserve  保留容量 可以扩容
    void reserve(size_t n)  //只是改变capacity 不改变size
    {
      if (n > _capacity)
      {
        //新建一个字符数组
        cout << "reserve->" << n << endl;
        char* new_str = new char[n + 1];
        //更改容量
        //strcpy(new_str, _str);
        memcpy(new_str, _str, _size + 1);
        delete[] _str;
        _str = new_str;
        _capacity = n;
      }
    }
    //push_back 
    void push_back(char ch)
    {
      if (_size == _capacity)
      {
        reserve(_capacity == 0 ? 4 : _capacity * 2);
      }
      //加入字符
      _str[_size] = ch;
      ++_size;
      _str[_size] = '\0';
    }
    void append(const char* str)
    {
      if (_size + strlen(str) > _capacity)
      {
        reserve(_size + strlen(str));//至少保留_size + strlen(str)
      }
      //加入字符串
      memcpy(_str + _size, str, strlen(str)+1);//在'\0'位置(就是_str末尾)+str
      _size += strlen(str);
    }
    //实现+=  也是使用push_back 和append函数
    string& operator+=(char ch)
    {
      push_back(ch);
      return *this;
    }
    string& operator+=(const char* str)
    {
      append(str);
      return *this;
    }
    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;
    }
    //从pos位置开始,删除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+len==_size
        //_str[end] = '\0';
        _size -= len;
      }
      return *this;
    }
  //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);
      if (str)
      {
        return str-_str;//两个指针相减,得到的是地址的偏移量
      }
      else {
        return npos;
      }
    }
    //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;
    }
    void clear()
    {
      _str[0] = '\0';
      _size = 0;
    }
    //实现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';
      }
    }
    //字符串比较按照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;
    }
    bool operator==(const string& s)
    {
      return _size == s._size && memcmp(_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);
    }
    //string& operator=(const string& s)
    //{
    //  if (this != &s)
    //  {
    //    //如果不是同一个string
    //    //深拷贝
    //    char* new_str = new char[s._capacity + 1];
    //    memcpy(new_str, s._str, s._size);
    //    //删除原来地址
    //    delete[] _str;
    //    //新指向一个new_str
    //    _str = new_str;
    //    //更改容量和size
    //    _capacity = s._capacity;
    //    _size = s._size;
    //  }
    //  return *this;
    //}
    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;
    }
    size_t capacity()
    {
      return _capacity;
    }
    size_t size()
    {
      return _size;
    }
    //无穷递归的问题:反复调用堆栈
    // std::swap(*this,tmp)
    //定义属性
  private:
    char* _str;
    int _size;
    int _capacity;
  public:
    const static size_t npos;
  };
  const size_t string::npos = -1;
}
//流提取
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;
}


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