string类的模拟实现(上)

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

string类的模拟实现

前文对于string的常用函数做了讲解,由于string是一个面试官常考的点,总喜欢让模拟实现string类,下面来模拟实现一下string,赋予基本的功能,且逐步完善函数实现方式。

string基本框架的实现

string类的基本框架,比如构造函数,拷贝构造,析构函数,成员变量,起码的push_back等一些能正常使得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(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;
    }
    //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;
    }
    //返回size
    size_t size() const  //const表示修饰this指针,也就是说只读,如果是const对象,也可以访问,普通用户相当于权限的缩小也可也访问
    {
      return _size;
    }
    size_t capacity()
    {
      return _capacity;
    }
  private:
    char* _str;
    int _size;
    int _capacity;
  };
}

构造函数和拷贝构造

默认构造函数和带参构造函数合并,使用缺省参数

拷贝构造函数,我们要使用深拷贝,因为如果是浅拷贝,仅仅是将数值传给新的string对象,但是两者对应一个地址一个空间,当析构一个string对象后,另一个对象再次析构就会报错。

string(const char* str = "")  //""字符串自带'\0'
  :_str(new char[strlen(str) + 1])   //因为我们底层用的数组,所以一定要多开一位空间存放'\0'
  , _size(strlen(str))
  , _capacity(strlen(str))
{
  //存储字符串
  strcpy(_str, str);  //传入字符串的时候,一般都是结尾为'\0',中间有'\0'的都是我们为string对象增加的。所以这个地方还是使用strcpy即可
}
//拷贝构造
string(const string& s)
{
  //深拷贝,就是创建一个大小一样的空间
  _str = new char[s._capacity + 1];
  strcpy(_str, s._str);
  _size = s._size;
  _capacity = s._capacity;
}

push_back和append的实现

想要实现push_back和append函数,都要在底层中考虑是否需要扩容,那么我们就顺势要写出reserve函数,让其来判断是否需要扩容。

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

注意:为什么拷贝字符串内容的时候用memcpy而不是strcpy,这是因为,string中可能中间会有'\0',memcpy是根据第三个参数来定要拷贝的字符长度,而strcmpy,是根据要拷贝的字符串的'\0'出现的位置,所以使用memcpy更加合适。

strcpy和memcpy的对比

  • char * strcpy ( char * destination, const char * source );
  • void * memcpy ( void * destination, const void * source, size_t num );
相关文章
|
2月前
|
编解码 Java 开发者
Java String类的关键方法总结
以上总结了Java `String` 类最常见和重要功能性方法。每种操作都对应着日常编程任务,并且理解每种操作如何影响及处理 `Strings` 对于任何使用 Java 的开发者来说都至关重要。
317 5
|
6月前
|
存储 编译器 C语言
关于string的‘\0‘与string,vector构造特点,反迭代器与迭代器类等的讨论
你真的了解string的'\0'么?你知道创建一个string a("abcddddddddddddddddddddddddd", 16);这样的string对象要创建多少个对象么?你知道string与vector进行扩容时进行了怎么的操作么?你知道怎么求Vector 最大 最小值 索引 位置么?
176 0
|
Java 索引
java基础(13)String类
本文介绍了Java中String类的多种操作方法,包括字符串拼接、获取长度、去除空格、替换、截取、分割、比较和查找字符等。
180 0
java基础(13)String类
|
9月前
|
缓存 安全 Java
《从头开始学java,一天一个知识点》之:字符串处理:String类的核心API
🌱 **《字符串处理:String类的核心API》一分钟速通!** 本文快速介绍Java中String类的3个高频API:`substring`、`indexOf`和`split`,并通过代码示例展示其用法。重点提示:`substring`的结束索引不包含该位置,`split`支持正则表达式。进一步探讨了String不可变性的高效设计原理及企业级编码规范,如避免使用`new String()`、拼接时使用`StringBuilder`等。最后通过互动解密游戏帮助读者巩固知识。 (上一篇:《多维数组与常见操作》 | 下一篇预告:《输入与输出:Scanner与System类》)
267 11
|
9月前
|
Java
课时14:Java数据类型划分(初见String类)
课时14介绍Java数据类型,重点初见String类。通过三个范例讲解:观察String型变量、&quot;+&quot;操作符的使用问题及转义字符的应用。String不是基本数据类型而是引用类型,但使用方式类似基本类型。课程涵盖字符串连接、数学运算与字符串混合使用时的注意事项以及常用转义字符的用法。
292 9
|
9月前
|
存储 JavaScript Java
课时44:String类对象两种实例化方式比较
本次课程的主要讨论了两种处理模式在Java程序中的应用,直接赋值和构造方法实例化。此外,还讨论了字符串池的概念,指出在Java程序的底层,DOM提供了专门的字符串池,用于存储和查找字符串。 1.直接赋值的对象化模式 2.字符串池的概念 3.构造方法实例化
188 1
|
Java
【编程基础知识】(讲解+示例实战)方法参数的传递机制(值传递及地址传递)以及String类的对象的不可变性
本文深入探讨了Java中方法参数的传递机制,包括值传递和引用传递的区别,以及String类对象的不可变性。通过详细讲解和示例代码,帮助读者理解参数传递的内部原理,并掌握在实际编程中正确处理参数传递的方法。关键词:Java, 方法参数传递, 值传递, 引用传递, String不可变性。
292 1
【编程基础知识】(讲解+示例实战)方法参数的传递机制(值传递及地址传递)以及String类的对象的不可变性
|
存储 编译器 C语言
【c++丨STL】string类的使用
本文介绍了C++中`string`类的基本概念及其主要接口。`string`类在C++标准库中扮演着重要角色,它提供了比C语言中字符串处理函数更丰富、安全和便捷的功能。文章详细讲解了`string`类的构造函数、赋值运算符、容量管理接口、元素访问及遍历方法、字符串修改操作、字符串运算接口、常量成员和非成员函数等内容。通过实例演示了如何使用这些接口进行字符串的创建、修改、查找和比较等操作,帮助读者更好地理解和掌握`string`类的应用。
460 2
String类-知识回顾①
这篇文章回顾了Java中String类的相关知识点,包括`==`操作符和`equals()`方法的区别、String类对象的不可变性及其好处、String常量池的概念,以及String对象的加法操作。文章通过代码示例详细解释了这些概念,并探讨了使用String常量池时的一些行为。
String类-知识回顾①
|
安全 Java 测试技术
Java零基础-StringBuffer 类详解
【10月更文挑战第9天】Java零基础教学篇,手把手实践教学!
400 2