C++深浅拷贝及简易string类实现方式

目录
  • 三种拷贝方式
    • 浅拷贝
    • 深拷贝
    • 写时拷贝
  • VS与GCC中的拷贝方式
    • Windows VS2022
    • Linux GCC
  • 简易string类
    • 传统版写法的string类
    • 现代版写法string类
  • 总结

三种拷贝方式

浅拷贝

对于自定义的string类,如果不显式定义拷贝构造函数,编译器会默认生成拷贝构造函数,此时的拷贝方式是浅拷贝,两个对象会公用一块儿内存,析构时同一空间被释放两次,会导致程序崩溃。

赋值运算符重载也会产生同样的问题,同时,由于被赋值对象原来有空间,浅拷贝还会导致旧的空间无法找到,造成内存泄漏。

深拷贝

类中设计到资源的管理,拷贝构造函数、赋值运算符重载以及析构函数都要显示给出,按照深拷贝的方式。

深拷贝的方式让每个对象都独立拥有一份资源,不会造成多次释放导致程序崩溃的问题。

写时拷贝

写时拷贝是通过浅拷贝+引用计数的方式来实现的,引用计数是用来记录资源的被引用的次数,

可以将这种写时拷贝的机制想象成“拖延症”,只有当不得不进行拷贝时,才会开辟新空间进行拷贝

VS与GCC中的拷贝方式

Windows VS2022

VS中采用的是深拷贝的方式

Linux GCC

GCC编译器采用的是写时拷贝的方式

简易string类

简易string类主要实现四个功能,即构造函数、拷贝构造函数、析构函数、赋值运算符重载,主要考察深浅拷贝

实现简易string类有两种代码风格,一种传统版写法,代码复用性第,可读性较好;另一种称为现代版写法,代码复用性高,但是较难理解。

传统版写法的string类

构造函数

步骤:

  • 判断是否为空指针,string类不允许nullptr构造对象
  • 申请新空间
  • 将字符串中的值拷贝到申请的空间
string(const char* str = "")
{
	if (nullptr == str)
	{
		assert(false);
		return;
	}
	//+1是因为有'\0',strcpy会将源字符串中的'\0'拷贝到目标空间
	_str = new char[strlen(str) + 1];
	strcpy(_str, str);
}

拷贝构造函数

步骤:

  • 开辟空间
  • 用源对象的_str给当前对象的_str赋值
string(const string& s)
	:_str(new char[strlen(s._str) + 1])
{
	strcpy(_str, s._str);
}

赋值运算符重载

步骤:

  • 判断是否自己给自己赋值
  • 开辟新空间
  • 拷贝元素
  • 删除旧空间
string& operator=(const string& s)
{
	//避免自己给自己赋值
	if (this != &s)
	{
		char* temp = new char[strlen(s._str) + 1];
		strcpy(temp, s._str);
		delete[] _str;
		_str = temp;
	}
	return *this;
}

另一种写法

这种写法不用定义临时变量,代码相对简洁一点,但是如果new申请空间失败,旧的空间也无法找到。

析构函数

步骤:

  • 释放空间
  • 将指针置为空
~string()
{
    if (_str)
    {
        delete[]_str;
        _str = nullptr;
    }
}

现代版写法string类

构造函数

string(const char* str = "")
{
	if (str == nullptr)
	{
		assert(false);
	}
	_str = new char[strlen(str) + 1];
	strcpy(_str, str);
}

拷贝构造函数

拷贝构造函数中利用构造函数,实现了代码的复用

步骤:

  • 在初始化列表中将_str置为空
  • 定义一个临时的string类对象,指向要拷贝的对象相同位置
  • 交换临时对象与当前对象的_str
string(const string& s)
	:_str(nullptr)
{
	//调用构造函数
	string temp(s._str);
	//交换以后temp指向空,函数退出后被销毁
	swap(_str, temp._str);
}

赋值运算符重载函数

步骤:

  • 判断是否为自己给自己赋值
  • 调用拷贝构造函数定义临时变量
  • 交换临时变量与当前对象的_str
string& operator=(string& s)
{
	if (this != &s)
	{
		string temp(s);
		swap(_str, s._str);
	}
	return *this;
}

更简洁的写法:

string& operator=(string s)
{
	//传参调用拷贝构造函数,不用判断是否给自己赋值
	swap(_str, s._str);
	return *this;
}

析构函数

~string()
{
	if (_str)
	{
		delete[] _str;
		_str = nullptr;
	}
}

总结

以上为个人经验,希望能给大家一个参考,也希望大家多多支持我们。

(0)

相关推荐

  • C++拷贝构造函数(深拷贝与浅拷贝)详解

    对于普通类型的对象来说,它们之间的复制是很简单的,例如:int a=88;int b=a; 而类对象与普通对象不同,类对象内部结构一般较为复杂,存在各种成员变量.下面看一个类对象拷贝的简单例子. 复制代码 代码如下: #include <iostream>using namespace std; class CExample {private:    int a;public:    CExample(int b)    { a=b;}    void Show ()    {       

  • C++深浅拷贝和string类的两种写法详解

    目录 一.深浅拷贝 二.string类的两种写法 1.传统写法 2.现代写法 总结 一.深浅拷贝 拷贝这个词对于我们来说应该不陌生,比如我们平常的复制和粘贴就是拷贝:但是如果把拷贝这个词放到C++中来说就有一些复杂了,我们先来看一下什么是浅拷贝: 下面用字符串类来模拟实现. class Astring { public: //构造函数 Astring(const char* str = "") { _str = new char[strlen(str) + 1]; strcpy(_st

  • C/C++ 浅拷贝和深拷贝的实例详解

    C/C++ 浅拷贝和深拷贝的实例详解 深拷贝是指拷贝对象的具体内容,而内存地址是自主分配的,拷贝结束之后,两个对象虽然存的值是相同的,但是内存地址不一样,两个对象也互不影响,互不干涉. 浅拷贝就是对内存地址的复制,让目标对象指针和源对象指向同一片内存空间. 浅拷贝只是对对象的简单拷贝,让几个对象共用一片内存,当内存销毁的时候,指向这片内存的几个指针需要重新定义才可以使用,要不然会成为野指针. 在iOS开发中也会涉及到浅拷贝和深拷贝,简而言之: 浅拷贝:拷贝指针变量的值 深拷贝:拷贝指针所指向内存

  • C++深浅拷贝及简易string类实现方式

    目录 三种拷贝方式 浅拷贝 深拷贝 写时拷贝 VS与GCC中的拷贝方式 Windows VS2022 Linux GCC 简易string类 传统版写法的string类 现代版写法string类 总结 三种拷贝方式 浅拷贝 对于自定义的string类,如果不显式定义拷贝构造函数,编译器会默认生成拷贝构造函数,此时的拷贝方式是浅拷贝,两个对象会公用一块儿内存,析构时同一空间被释放两次,会导致程序崩溃. 赋值运算符重载也会产生同样的问题,同时,由于被赋值对象原来有空间,浅拷贝还会导致旧的空间无法找到

  • String类的写时拷贝实例

    实例如下: #include<iostream> using namespace std; class String; ostream& operator<<(ostream &out, const String&s); //引用计数器类 class String_rep { friend class String; friend ostream& operator<<(ostream &out, const String&

  • 详解iOS的深浅拷贝

    前言 OC对象的三种拷贝方式 OC的对象拷贝有如下三种方式,很多时候我们把深复制和完全复制混为一谈,其他他们是有区别的,具体如下 浅复制(shallow copy):在浅复制操作时,对于被复制对象的每一层都是指针复制. 深复制(one-level-deep copy):在深复制操作时,对于被复制对象,至少有一层是深复制. 完全复制(real-deep copy):在完全复制操作时,对于被复制对象的每一层都是对象复制. 两图以避之 理解深复制(mutableCopy) 浅复制很简单,就不演示了,看

  • c++中深浅拷贝以及写时拷贝的实现示例代码

    本文主要给大家介绍了关于c++中深浅拷贝及写时拷贝实现的相关内容,分享出来供大家参考学习,下面话不多说,来一起看看详细的介绍: 一:浅拷贝&深拷贝 浅拷贝:在拷贝构造的时候,直接将原内容的地址交给要拷贝的类,两个类共同指向一片空间.但是存在很大的缺陷:①一旦对s2进行操作,s1的内容也会改变:②析构时先析构s2,再析构s1,但是由于s1,s2指向同一片空间,会导致一片空间的二次析构导致出错. 深拷贝:通过开辟和源空间大小相同的空间并将内容拷贝下来再进行操作.不论是否对s2进行操作,都会拷贝一片相

  • C#深浅拷贝的深入解析

    前言 前面我们学习完了设计模式,在其中我们有了解到原型模式.这里涉及到了克隆自身对象.那么也就是对对象进行拷贝.这里就涉及到了这么一个概念.深浅拷贝.何为深拷贝何为浅拷贝呢?我们一起来看看吧. 浅拷贝 首先我们看看浅拷贝.浅拷贝就是将对象中的所有字段复制到新对象中去,浅拷贝对于值类型和引用类型有不同的影响.值类型的值被复制到副本中后,修改副本中的值不会影响原来对象的值.然而引用类型被复制到副本中的是引用类型的引用.不是引用的对象.这样再修改副本中的值是会导致原来对象的值也被修改了.但是这里引用类

  • 探讨Java中的深浅拷贝问题

    一.前言 拷贝这个词想必大家都很熟悉,在工作中经常需要拷贝一份文件作为副本.拷贝的好处也很明显,相较于新建来说,可以节省很大的工作量.在Java中,同样存在拷贝这个概念,拷贝的意义也是可以节省创建对象的开销. Object类中有一个方法clone(),具体方法如下: protected native Object clone() throws CloneNotSupportedException; 1.该方法由 protected 修饰,java中所有类默认是继承Object类的,重载后的clo

  • C++中的string类的用法小结

    相信使用过MFC编程的朋友对CString这个类的印象应该非常深刻吧?的确,MFC中的CString类使用起来真的非常的方便好用.但是如果离开了MFC框架,还有没有这样使用起来非常方便的类呢?答案是肯定的.也许有人会说,即使不用MFC框架,也可以想办法使用MFC中的API,具体的操作方法在本文最后给出操作方法.其实,可能很多人很可能会忽略掉标准C++中string类的使用.标准C++中提供的string类得功能也是非常强大的,一般都能满足我们开发项目时使用.现将具体用法的一部分罗列如下,只起一个

  • 详解C++中String类模拟实现以及深拷贝浅拷贝

    详解C++中String类模拟实现以及深拷贝浅拷贝 在C语言中/C++中,字符串是一个应用很广泛的类型,也是很基础的类型,C语言并没有直接处理字符串的操作而是采用字符指针和字符串数组进行操作,而在C++中标准库为我们封装了一个字符串的类供我们使用,使用需要#inlcude <string>头文件.我们也可以自己模拟实现一个简单的String类. 在模拟实现String类的过程中,不可避免的会遇到深拷贝浅拷贝的问题,下面就深拷贝浅拷贝做一个简介.所谓深拷贝浅拷贝,简单来说就是浅拷贝只是简单的将值

  • 浅谈js基础数据类型和引用类型,深浅拷贝问题,以及内存分配问题

    js 深浅拷贝问题 浅拷贝一般指的是基本类型的复制 深拷贝一般指引用类型的拷贝,把引用类型的值也拷贝出来 举例 h5的sessionStorage只能存放字符串,所以要存储json时就要把json使用JSON.stringify()把json转换成string,然后再用JSON.parse()转换成json数据 缺点:JSON.parse和JSON.stringify只支持IE9+以上 解决这个问题可以使用深度比那里拷贝方法 js 中内存分配问题(堆和栈) js中基本类型类型一般是存储在栈中的.

随机推荐