C++实现一个简单的String类

本文涉及的产品
实时计算 Flink 版,5000CU*H 3个月
实时数仓Hologres,5000CU*H 100GB 3个月
检索分析服务 Elasticsearch 版,2核4GB开发者规格 1个月
简介: 使用基本的C++知识实现一个简单的String类,这个类中包含了C++常用的知识点。感觉是很有意思的一个小代码片段。跟大家分享一下我的实现,欢迎大家批评指正。

C++实现一个简单的String类

使用基本的C++知识实现一个简单的String类,这个类中包含了C++常用的知识点。感觉是很有意思的一个小代码片段。

跟大家分享一下我的实现,欢迎大家批评指正。

类声明

  1. 该类中包含了三个构造函数:有参数的构造函数,拷贝构造函数已经移动构造函数

  2. 重载了[],=(一个普通赋值运算符,一个移动赋值运算符),+,==四个运算符

  3. 一个用于求字符长度的方法;一个用于获取C语言类型字符串的方法

  4. 以友元的方式重载了输入流>>和输出流<<操作符

头文件(strings.h)

//
// Created by Zhenyu Tan on 2018/10/3.
//
#include <iostream>

class String {
private:
    char* _buffer;
    size_t _length;
    void init(const char* str);

public:
    String(const char* str= nullptr);       // 默认构造函数
    String(const String& other);            // 拷贝构造函数
    String(String&& other) noexcept;        // 移动构造函数
    ~String();                              // 析构函数

    size_t length();
    const char* data();

    char& operator[](size_t index);
    String& operator=(const String& other);
    String& operator=(String&& other) noexcept;
    String operator+(const String& other);
    bool operator==(const String& other);

    friend std::ostream& operator<<(std::ostream& output, const String& str);
    friend std::istream& operator>>(std::istream& input, String& str);
};

类实现

源文件(strings.cpp)

//
// Created by Zhenyu Tan on 2018/10/5.
//

#include "strings.h"
#include <cstring>
#include <exception>
#include <iostream>

using std::cout;
using std::ostream;
using std::istream;

size_t String::length() {
    if (0 == _length) {
        _length = std::strlen(_buffer);
    }
    return _length;
}

const char* String::data() {
    return _buffer;
}


void String::init(const char* str) {
    if (nullptr == str) {
        _length = 0;
        _buffer = nullptr;
    } else {
        _length = std::strlen(str);
        _buffer = new char[_length + 1];
        std::strcpy(_buffer, str);
    }
}


String::String(const char* str) {
    init(str);
    cout << "默认构造函数(" << *this << ")\n";
}


String::String(const String& other) {
    // 在类的成员函数中可以访问同类型实例的私有变量
    init(other._buffer);
    cout << "拷贝构造函数(" << *this << ")\n";
}

String::String(String&& other) noexcept {
    // 把other对象掏空用来填充this
    _buffer = nullptr;
    _buffer = other._buffer;
    _length = other._length;
    other._buffer = nullptr;
    other._length = 0;
    cout << "移动构造函数(" << *this << ")\n";
}


String::~String() {
    delete[] _buffer;
    cout << "析构函数(" << *this << ")\n";
}

/*
 * 拷贝构造函数使用传入对象的值生成一个新的对象的实例
 * 赋值运算符是将对象的值复制给一个已经存在的实例
 */
String& String::operator=(const String& other) {
    if (this != &other) {
        delete[] _buffer;
        init(other._buffer);
    }
    cout << "拷贝赋值操作(" << *this << ")\n";
    return *this;
}

/*
 * 移动赋值操作即把参数传进来的对象的所有权转移到this指向的对象
 * 掏空other对象的所有
 */
String& String::operator=(String&& other) noexcept {
   if (this != &other) {
       _buffer = nullptr;
       _buffer = other._buffer;
       _length = other._length;
       other._buffer = nullptr;
       other._length = 0;
   }
    cout << "移动赋值操作(" << *this << ")\n";
    return *this;
}


char& String::operator[](size_t index) {
    if (index >= _length) {
        throw std::out_of_range("Index out of range");
    } else {
        return _buffer[index];
    }
}


bool String::operator==(const String& other) {
    if (_length != other._length) {
        return false;
    } else {
        return 0 == std::strcmp(_buffer, other._buffer);
    }
}

/*
 * 关于是返回对象本身还是返回对象引用
 * 如果函数返回在函数中创建的临时对象,则不要使用引用
 * 如果函数返回的是通过引用或指针传递给它的对象,则应当按引用返回对象
 * 如果先创建一个对象,然后返回改对象的副本,则可以使用返回对象
 */
String String::operator+(const String& other) {
    String _str;
    if (nullptr == _buffer) {
        _str = other;
    } else if (nullptr == other._buffer) {
        _str = *this;
    } else {
        _str._buffer = new char[_length + other._length + 1];
        std::strcpy(_str._buffer, _buffer);
        std::strcat(_str._buffer, other._buffer);
        _str._length = std::strlen(_str._buffer);
    }
    return _str;
}


ostream& operator<<(ostream &output, const String& str) {
    if (nullptr == str._buffer) {
        output << "";
    } else {
        output << str._buffer;
    }
    return output;
}

istream& operator>>(istream &input, String& str) {
    input >> str._buffer;
    return input;
}

调用示例

#include "strings.h"
#include <iostream>

using std::cout;

int main() {
    String str1("Hello");
    cout << str1.data() << '\n';
    cout << str1.length() << '\n';
    cout << "----------\n";
    String str2 = "Word";
    cout << str2 << '\n';
    cout << "----------\n";
    String str3 = str1 + str2;
    cout << str3.data() << '\n';
    cout << str3.length() << '\n';
    return 0;
}

运行结果:

默认构造函数(Hello)
Hello
5
----------
默认构造函数(Word)
Word
----------
默认构造函数()
HelloWord
9
析构函数(HelloWord)
析构函数(Word)
析构函数(Hello)

主程序中的第7行和第11行各自调用一次默认的有参构造函数,第14行是重载的加法运算符中调用了一次无参的构造函数(由于C++编译器的优化,函数返回值没有调用拷贝构造函数)

目录
相关文章
|
2天前
|
编译器 C语言 C++
类和对象的简述(c++篇)
类和对象的简述(c++篇)
|
2天前
|
C++
模拟实现c++中的string
模拟实现c++中的string
|
1月前
|
C++ 芯片
【C++面向对象——类与对象】Computer类(头歌实践教学平台习题)【合集】
声明一个简单的Computer类,含有数据成员芯片(cpu)、内存(ram)、光驱(cdrom)等等,以及两个公有成员函数run、stop。只能在类的内部访问。这是一种数据隐藏的机制,用于保护类的数据不被外部随意修改。根据提示,在右侧编辑器补充代码,平台会对你编写的代码进行测试。成员可以在派生类(继承该类的子类)中访问。成员,在类的外部不能直接访问。可以在类的外部直接访问。为了完成本关任务,你需要掌握。
68 19
|
1月前
|
存储 编译器 数据安全/隐私保护
【C++面向对象——类与对象】CPU类(头歌实践教学平台习题)【合集】
声明一个CPU类,包含等级(rank)、频率(frequency)、电压(voltage)等属性,以及两个公有成员函数run、stop。根据提示,在右侧编辑器补充代码,平台会对你编写的代码进行测试。​ 相关知识 类的声明和使用。 类的声明和对象的声明。 构造函数和析构函数的执行。 一、类的声明和使用 1.类的声明基础 在C++中,类是创建对象的蓝图。类的声明定义了类的成员,包括数据成员(变量)和成员函数(方法)。一个简单的类声明示例如下: classMyClass{ public: int
50 13
|
1月前
|
编译器 数据安全/隐私保护 C++
【C++面向对象——继承与派生】派生类的应用(头歌实践教学平台习题)【合集】
本实验旨在学习类的继承关系、不同继承方式下的访问控制及利用虚基类解决二义性问题。主要内容包括: 1. **类的继承关系基础概念**:介绍继承的定义及声明派生类的语法。 2. **不同继承方式下对基类成员的访问控制**:详细说明`public`、`private`和`protected`继承方式对基类成员的访问权限影响。 3. **利用虚基类解决二义性问题**:解释多继承中可能出现的二义性及其解决方案——虚基类。 实验任务要求从`people`类派生出`student`、`teacher`、`graduate`和`TA`类,添加特定属性并测试这些类的功能。最终通过创建教师和助教实例,验证代码
52 5
|
1月前
|
存储 算法 搜索推荐
【C++面向对象——群体类和群体数据的组织】实现含排序功能的数组类(头歌实践教学平台习题)【合集】
1. **相关排序和查找算法的原理**:介绍直接插入排序、直接选择排序、冒泡排序和顺序查找的基本原理及其实现代码。 2. **C++ 类与成员函数的定义**:讲解如何定义`Array`类,包括类的声明和实现,以及成员函数的定义与调用。 3. **数组作为类的成员变量的处理**:探讨内存管理和正确访问数组元素的方法,确保在类中正确使用动态分配的数组。 4. **函数参数传递与返回值处理**:解释排序和查找函数的参数传递方式及返回值处理,确保函数功能正确实现。 通过掌握这些知识,可以顺利地将排序和查找算法封装到`Array`类中,并进行测试验证。编程要求是在右侧编辑器补充代码以实现三种排序算法
40 5
|
1月前
|
Serverless 编译器 C++
【C++面向对象——类的多态性与虚函数】计算图像面积(头歌实践教学平台习题)【合集】
本任务要求设计一个矩形类、圆形类和图形基类,计算并输出相应图形面积。相关知识点包括纯虚函数和抽象类的使用。 **目录:** - 任务描述 - 相关知识 - 纯虚函数 - 特点 - 使用场景 - 作用 - 注意事项 - 相关概念对比 - 抽象类的使用 - 定义与概念 - 使用场景 - 编程要求 - 测试说明 - 通关代码 - 测试结果 **任务概述:** 1. **图形基类(Shape)**:包含纯虚函数 `void PrintArea()`。 2. **矩形类(Rectangle)**:继承 Shape 类,重写 `Print
48 4
|
1月前
|
设计模式 IDE 编译器
【C++面向对象——类的多态性与虚函数】编写教学游戏:认识动物(头歌实践教学平台习题)【合集】
本项目旨在通过C++编程实现一个教学游戏,帮助小朋友认识动物。程序设计了一个动物园场景,包含Dog、Bird和Frog三种动物。每个动物都有move和shout行为,用于展示其特征。游戏随机挑选10个动物,前5个供学习,后5个用于测试。使用虚函数和多态实现不同动物的行为,确保代码灵活扩展。此外,通过typeid获取对象类型,并利用strstr辅助判断类型。相关头文件如&lt;string&gt;、&lt;cstdlib&gt;等确保程序正常运行。最终,根据小朋友的回答计算得分,提供互动学习体验。 - **任务描述**:编写教学游戏,随机挑选10个动物进行展示与测试。 - **类设计**:基类
32 3
|
3月前
|
C语言 C++ 容器
【c++丨STL】string模拟实现(附源码)
本文详细介绍了如何模拟实现C++ STL中的`string`类,包括其构造函数、拷贝构造、赋值重载、析构函数等基本功能,以及字符串的插入、删除、查找、比较等操作。文章还展示了如何实现输入输出流操作符,使自定义的`string`类能够方便地与`cin`和`cout`配合使用。通过这些实现,读者不仅能加深对`string`类的理解,还能提升对C++编程技巧的掌握。
140 5
|
3月前
|
存储 编译器 C语言
【c++丨STL】string类的使用
本文介绍了C++中`string`类的基本概念及其主要接口。`string`类在C++标准库中扮演着重要角色,它提供了比C语言中字符串处理函数更丰富、安全和便捷的功能。文章详细讲解了`string`类的构造函数、赋值运算符、容量管理接口、元素访问及遍历方法、字符串修改操作、字符串运算接口、常量成员和非成员函数等内容。通过实例演示了如何使用这些接口进行字符串的创建、修改、查找和比较等操作,帮助读者更好地理解和掌握`string`类的应用。
89 2