C++自定义String类

自定义一个String类型,该类包含一个指向字符串的指针和一个统计对象数量的计数器.

代码如下:

cpp 复制代码
//string.h
#pragma once
//String类型
#include <iostream>
using namespace std;

class String
{
private:
    char* m_str;//保存字符串的地址
    static int num_strings;//创建的对象数量
public:
    String(const char* str = NULL);//构造函数
    ~String();//析构函数
    String(const String& s);//拷贝构造函数
    String& operator=(const String& s);//=重载函数
    friend ostream& operator<<(ostream & os,const String &s);//重载<<
};
cpp 复制代码
//string.cpp
#define _CRT_SECURE_NO_WARNINGS
#include "string.h"
#include <cstring>

int String::num_strings = 0;//类中的变量

String::String(const char* str)//构造函数
{
    if (str == NULL)
    {
        m_str = new char[strlen("趣字节") + 1];
        strcpy(m_str,"趣字节");
    }
    else
    {
        m_str = new char[strlen(str) + 1];
        strcpy(m_str,str);
    }
    ++num_strings;
    cout << "构造函数:" << m_str << ",对象数量:" << num_strings << endl;//不重要的输出
}
String::~String() //析构函数
{
    delete[]m_str; 
    m_str = NULL;
    --num_strings;
    cout << "析构函数:"  << "对象数量:" << num_strings << endl;//不重要的输出
}

String::String(const String& s)//拷贝构造函数
{
    m_str = new char[strlen(s.m_str) + 1];
    strcpy(m_str,s.m_str);
    ++num_strings;
    cout << "拷贝构造函数:" << m_str << ",对象数量:" << num_strings << endl;//不重要的输出
}
String& String::operator=(const String& s)//=重载函数
{
    if (&s == this)
        return *this;
    delete[]m_str;
    m_str = new char[strlen(s.m_str)+1];
    strcpy(m_str,s.m_str);
    cout << "=赋值函数:" << m_str << ",对象数量:" << num_strings << endl;//不重要的输出
    return *this;
}

ostream& operator<<(ostream& os, const String& s)//重载<<
{
    os << s.m_str << endl;
    return os;
}
cpp 复制代码
//test.cpp测试代码
#include"string.h"

int main()
{
    String s1("趣字节,有趣的编程!");

    String s2(s1);
    String s3 = s1;
    String s4 = String(s1);
    String* ps4 = new String(s1);
    String s5;
    s5 = s1;

    delete ps4;

    return 0;
}

何时调用拷贝(复制)构造函数?下面4中情况都将调用拷贝构造函数:

cpp 复制代码
String s2(s1);                 //1.调用拷贝构造函数
String s3 = s1;                //2.调用拷贝构造函数
String s4 = String(s1);        //3.调用拷贝构造函数
String* ps4 = new String(s1);  //4.调用拷贝构造函数

第1种情况是调用拷贝构造函数的典型写法;

第2种情况不是先定义s3然后再把s1赋值给s3,在初始化阶段这两步合并成拷贝构造函数;注意和s5的区别。

第3种情况,可能调用拷贝构造函数直接创建s4,也可能使用拷贝构造函数生成一个临时对象,然后再赋值给s4,这取决于编译器的具体实现,但我们使用的编译器都是直接调用拷贝构造函数直接创建s4;

第4种,调用拷贝构造创建一个匿名对象,然后把对象的地址赋给ps4指针。

ps4是通过new创建的,在程序结束前必须调用delete释放,delete ps4时会调用ps4的析构函数。

s1,s2,s3,s3,s5都是局部变量,当main函数结束时生命周期自动结束,分别调用自己的析构函数。由于局部变量在栈中,栈的特点是后进先出,所以析构的顺序是s5,s4,s3,s2,s1。

相关推荐
虚拟之15 分钟前
36、stringstream
c++
我很好我还能学20 分钟前
【面试篇 9】c++生成可执行文件的四个步骤、悬挂指针、define和const区别、c++定义和声明、将引用作为返回值的好处、类的四个缺省函数
开发语言·c++
蓝婷儿41 分钟前
6个月Python学习计划 Day 16 - 面向对象编程(OOP)基础
开发语言·python·学习
渣渣盟1 小时前
基于Scala实现Flink的三种基本时间窗口操作
开发语言·flink·scala
糯米导航1 小时前
Java毕业设计:办公自动化系统的设计与实现
java·开发语言·课程设计
糯米导航1 小时前
Java毕业设计:WML信息查询与后端信息发布系统开发
java·开发语言·课程设计
南岩亦凛汀2 小时前
在Linux下使用wxWidgets进行跨平台GUI开发
c++·跨平台·gui·开源框架·工程实战教程
MessiGo2 小时前
Javascript 编程基础(5)面向对象 | 5.1、构造函数实例化对象
开发语言·javascript·原型模式
大霞上仙2 小时前
nonlocal 与global关键字
开发语言·python
曦月逸霜2 小时前
第34次CCF-CSP认证真题解析(目标300分做法)
数据结构·c++·算法