【c++】string类的模拟实现--1

简介: 【c++】string类的模拟实现--1

介绍:

       STL容器的模拟实现是我们了解STL函数接口的重要部分,也是在学习C++容器时的重要环节。这里,我们模拟实现string的常规接口。


steing的框架结构:


       在调试窗口上可观察到string的参数有size(大小)、capacity(大小)、串(char*)。至于迭代器,这里可先认为指针。


//框架设计模型
class string
{
public:
    typedef char* iterator;   //迭代器
private:
    char* _str;               //串
    size_t _capacity;    //容量
    size_t _size;           //大小
};


一,构造函数和析构函数

       构造函数在实现时首先要注意缺省值的情况。当创建string空类时,默认不会初始化,且输出时不会输出乱码,无任何数据,因此,这里需将缺省值设为""。


//String(const char* str = "\0") 错误示范,可能会出现程序崩溃
//String(const char* str = nullptr) 错误示范,可能会出现程序崩溃
//普通构造
string(const char* str = "")
     : _capacity(strlen(str) + 1)
     , _size(strlen(str))
{
     // 构造String类对象时,如果传递nullptr指针,可以认为程序非
     if (nullptr == str)
     {
         assert(false);
         return;
     }
     _str = new char[_capacity];
     memcpy(_str, str, _capacity);
}
//拷贝构造
string(const string& s)
    : _capacity(s._capacity)
    , _size(s._size)
{
    // 构造String类对象时,如果传递nullptr指针,可以认为程序非
    if (nullptr == s)
    {
        assert(false);
        return;
    }
    _str = new char[_capacity];
    memcpy(_str, s._str, s._size + 1);
}
//析构函数
~string()
{
      delete[] _str;
      _str = nullptr;
      _capacity = _size = 0;
}


二,赋值运算符与流运算符

       赋值运算符重载跟构造函数的注意事项一样,需注意的是这里进行的是深拷贝,不是浅拷贝,即重新开辟新的空间进行拷贝。


string& operator=(const string& s)
{
    delete[] _str;
    char* str = new char[s._capacity];
    memcpy(str, s._str, s._size + 1);
    _str = str;
    _size = s._size;
    _capacity = s._capacity;
    return *this;
}


       流的操作符中,输出流实现简单,只需根据string类的大小进行输出字符即可,但输入流需注意以下两点:


在进行对string输入时,若里面有数据,需将其清空。

string类的输入操作可能会影响容量的变化和大小的变化,若容量不够需进行扩容。

//输出流
ostream& operator<<(ostream& _cout, const bit::string& s)
{
    for (int i = 0; i < s._size; i++)
    {
        _cout << s._str[i];
    }
    return _cout;
}
//输入流
istream& operator>>(istream& _cin, bit::string& s)
{
    //先清理数据string中的数据
    delete[] s._str;
    char* str = new char[s._capacity];
    s._str = str;
    memcpy(s._str, "\0", 1);
    s._size = 0;
    //下面进行开始输入操作
    char buff[128];
    char ch = _cin.get();
    int i = 0;
    while (ch != ' ' && ch != '\n')
    {
        buff[i++] = ch;
        if (i == 127)
        {
            buff[i] = '\0';
            //在进行增添之前要先判断容量是否够容载
            if (s._capacity < s._size + strlen(buff))
            {
                s._capacity += 2 * (s._size + strlen(buff));
                char* str = new char[s._capacity];
                memcpy(str, s._str, s._size + 1);
                delete[] s._str;
                s._str = str;
            }
            //连接操作,注意:这里不能用strcat,因为strcat本身有bug,如下
           /*char str[] = "\0";
            strcat(str, "avbsc");
            cout << str << endl;
            此程序将会崩溃,strcat的第一个参数不能为空*/
            int k = 0;
            for (int j = s._size; j < s._size + strlen(buff); j++)
            {
                s._str[j] = buff[k++];
            }
            s._size += strlen(buff);
            i = 0;
        }
        ch = _cin.get();
    }
    buff[i] = '\0';
    //判断容量
    if (s._capacity < s._size + strlen(buff))
    {
        s._capacity += 2 * (s._size + strlen(buff));
        char* str = new char[s._capacity];
        memcpy(str, s._str, s._size + 1);
        delete[] s._str;
        s._str = str;
    }
    //连接操作
    int k = 0;
    for (int j = s._size; j < s._size + strlen(buff); j++)
    {
        s._str[j] = buff[k++];
    }
    s._size += strlen(buff);
    return _cin;
}

三,迭代器和运算符重载

       我们目前最常用的迭代器是begin()和end(),begin()指向首元素的地址,而end()指向尾元素的下一个地址处,实现机制如下:


iterator begin()
{
    return _str;
}
iterator end()
{
    return _str + _size;
}


       至于运算符重载的实现,这里,我们对 “[]、>、>=、<、<=、==、!=” 进行重载实现。


char& operator[](size_t index)
{
    //防止遍历出界
    assert(index <= _size && index >= 0);
    return *(_str + index);
}
const char& operator[](size_t index)const
{
    //防止遍历出界
    assert(index <= _size && index >= 0);
    return (const char)(*_str + index);
}
bool operator<(const string& s)
{
    for (int i = 0; i < _size; i++)
    {
        if (_str[i] < s._str[i])
        {
            return true;
        }
        else if (_str[i] > s._str[i])
        {
            return false;
        }
    }
    return false;
}
bool operator<=(const string& s)
{
    for (int i = 0; i < _size; i++)
    {
        if (_str[i] < s._str[i])
        {
            return true;
        }
        else if (_str[i] > s._str[i])
        {
            return false;
        }
    }
    return true;
}
bool operator>(const string& s)
{
    return !(*this <= s);
}
bool operator>=(const string& s)
{
    return !(*this < s);
}
bool operator==(const string& s)
{
    return !(*this > s) && !(*this < s);
}
bool operator!=(const string& s)
{
    return !(*this == s);
}


【c++】string类的模拟实现--2 https://developer.aliyun.com/article/1424722?spm=a2c6h.13148508.setting.24.214f4f0elA5UEs

相关文章
|
21天前
|
C语言 C++ 容器
【c++丨STL】string模拟实现(附源码)
本文详细介绍了如何模拟实现C++ STL中的`string`类,包括其构造函数、拷贝构造、赋值重载、析构函数等基本功能,以及字符串的插入、删除、查找、比较等操作。文章还展示了如何实现输入输出流操作符,使自定义的`string`类能够方便地与`cin`和`cout`配合使用。通过这些实现,读者不仅能加深对`string`类的理解,还能提升对C++编程技巧的掌握。
46 5
|
21天前
|
存储 编译器 C语言
【c++丨STL】string类的使用
本文介绍了C++中`string`类的基本概念及其主要接口。`string`类在C++标准库中扮演着重要角色,它提供了比C语言中字符串处理函数更丰富、安全和便捷的功能。文章详细讲解了`string`类的构造函数、赋值运算符、容量管理接口、元素访问及遍历方法、字符串修改操作、字符串运算接口、常量成员和非成员函数等内容。通过实例演示了如何使用这些接口进行字符串的创建、修改、查找和比较等操作,帮助读者更好地理解和掌握`string`类的应用。
31 2
|
27天前
|
存储 编译器 C++
【c++】类和对象(下)(取地址运算符重载、深究构造函数、类型转换、static修饰成员、友元、内部类、匿名对象)
本文介绍了C++中类和对象的高级特性,包括取地址运算符重载、构造函数的初始化列表、类型转换、static修饰成员、友元、内部类及匿名对象等内容。文章详细解释了每个概念的使用方法和注意事项,帮助读者深入了解C++面向对象编程的核心机制。
70 5
|
1月前
|
存储 编译器 C++
【c++】类和对象(中)(构造函数、析构函数、拷贝构造、赋值重载)
本文深入探讨了C++类的默认成员函数,包括构造函数、析构函数、拷贝构造函数和赋值重载。构造函数用于对象的初始化,析构函数用于对象销毁时的资源清理,拷贝构造函数用于对象的拷贝,赋值重载用于已存在对象的赋值。文章详细介绍了每个函数的特点、使用方法及注意事项,并提供了代码示例。这些默认成员函数确保了资源的正确管理和对象状态的维护。
72 4
|
1月前
|
存储 编译器 Linux
【c++】类和对象(上)(类的定义格式、访问限定符、类域、类的实例化、对象的内存大小、this指针)
本文介绍了C++中的类和对象,包括类的概念、定义格式、访问限定符、类域、对象的创建及内存大小、以及this指针。通过示例代码详细解释了类的定义、成员函数和成员变量的作用,以及如何使用访问限定符控制成员的访问权限。此外,还讨论了对象的内存分配规则和this指针的使用场景,帮助读者深入理解面向对象编程的核心概念。
82 4
|
2月前
|
Java
【编程基础知识】(讲解+示例实战)方法参数的传递机制(值传递及地址传递)以及String类的对象的不可变性
本文深入探讨了Java中方法参数的传递机制,包括值传递和引用传递的区别,以及String类对象的不可变性。通过详细讲解和示例代码,帮助读者理解参数传递的内部原理,并掌握在实际编程中正确处理参数传递的方法。关键词:Java, 方法参数传递, 值传递, 引用传递, String不可变性。
65 1
【编程基础知识】(讲解+示例实战)方法参数的传递机制(值传递及地址传递)以及String类的对象的不可变性
|
2月前
|
安全 Java 测试技术
Java零基础-StringBuffer 类详解
【10月更文挑战第9天】Java零基础教学篇,手把手实践教学!
45 2
|
2月前
|
存储 编译器 对象存储
【C++打怪之路Lv5】-- 类和对象(下)
【C++打怪之路Lv5】-- 类和对象(下)
31 4
|
2月前
|
编译器 C语言 C++
【C++打怪之路Lv4】-- 类和对象(中)
【C++打怪之路Lv4】-- 类和对象(中)
26 4
|
2月前
|
存储 安全 C++
【C++打怪之路Lv8】-- string类
【C++打怪之路Lv8】-- string类
26 1