深拷贝与浅拷贝,还有一道比较好的面试题

转自如下:

https://blog.csdn.net/zzwdkxx/article/details/53409803

==================================================

1. 深拷贝和浅拷贝(拷贝构造函数的使用)

有时候需要自己定义拷贝构造函数,以避免浅拷贝问题。

在什么情况下需要用户自己定义拷贝构造函数:

一般情况下,当类中成员有指针变量、类中有动态内存分配时常常需要用户自己定义拷贝构造函数。

在什么情况下系统会调用拷贝构造函数:(三种情况)

(1)用类的一个对象去初始化另一个对象时

(2)当函数的形参是类的对象时(也就是值传递时),如果是引用传递则不会调用

(3)当函数的返回值是类的对象或引用时

简单示例:

#include <iostream>
using namespace std;  

class A
{
private:
    int a;
public:
    A(int i){a=i;}  //内联的构造函数
    A(A &aa);
    int geta(){return a;}
};  

A::A(A &aa)     //拷贝构造函数
{
    a=aa.a;
    cout<<"拷贝构造函数执行!"<<endl;
}  

int get_a(A aa)     //参数是对象,是值传递,会调用拷贝构造函数
{
    return aa.geta();
}  

int get_a_1(A &aa)  //如果参数是引用类型,本身就是引用传递,所以不会调用拷贝构造函数
{
    return aa.geta();
}  

A get_A()       //返回值是对象类型,会调用拷贝构造函数。会调用拷贝构造函数,因为函数体内生成的对象aa是临时的,离开这个函数就消失了。所有会调用拷贝构造函数复制一份。
{
    A aa(1);
    return aa;
}  

A& get_A_1()    //会调用拷贝构造函数,因为函数体内生成的对象aa是临时的,离开这个函数就消失了。所有会调用拷贝构造函数复制一份。
{
    A aa(1);
    return aa;
}  

int _tmain(int argc, _TCHAR* argv[])
{
    A a1(1);
    A b1(a1);           //用a1初始化b1,调用拷贝构造函数
    A c1=a1;            //用a1初始化c1,调用拷贝构造函数  

    int i=get_a(a1);        //函数形参是类的对象,调用拷贝构造函数
    int j=get_a_1(a1);      //函数形参类型是引用,不调用拷贝构造函数  

    A d1=get_A();       //调用拷贝构造函数
    A e1=get_A_1();     //调用拷贝构造函数  

    return 0;
}  

附:一个面试试题

修改下面程序中的错误:

#include <iostream>
using namespace std;  

class NameStr
{
private:
    char *m_pName;
    char *m_pData;
public:
    NameStr()
    {
        static const char s_szDefaultName[]="Default name";
        static const char s_szDefaultStr[]="Default string";
        strcpy(m_pName,s_szDefaultName);
        strcpy(m_pData,s_szDefaultStr);
    }
    ~NamedStr(){}
    NameStr(const char* pName,const char* pData)
    {
        m_pData=new char[strlen(pData)];
        m_pName=new char[strlen(pData)];
    }  

    void Print()
    {
        cout<<"Name:"<<m_pName<<endl;
        cout<<"String:"<<m_pData<<endl;
    }
};  

int _tmain(int argc, _TCHAR* argv[])
{
    NameStr* pDefNss=NULL;  

    pDefNss=new NameStr[10];
    NameStr ns("hello","world");  

    delete pDefNss;  

    return 0;
}  

分析:

1. 第14、15行,strcpy(m_pName,s_szDefaultName) 对未分配内存空间的字符指针赋值会出现异常。

2. 第20行、21行,m_pData=new char[strlen(pData)] 应该为m_pData=new char[strlen(pData)+1] ,并且应该为最后一个字符赋值为‘\0‘。

3. 析构函数中,应该处理字符指针内存空间的释放。

4. 因为类的成员变量中有指针变量,因此应该编写类的拷贝构造函数和赋值函数,防止浅拷贝。

5. pDefNss是一个对象数组,delete时应该是delete [ ]pDefNss。

比较规范的代码如下:

#include <iostream>
using namespace std;  

//NameStr类的声明
class NameStr
{
private:
    char *m_pName;
    char *m_pData;
public:
    NameStr();      //默认拷贝构造函数  

    ~NameStr(); //析构函数声明  

    NameStr(const char* pName,const char* pData);   //带参构造函数的声明  

    NameStr(const NameStr& temp);   //拷贝构造函数的声明  

    NameStr& operator= (const NameStr& temp);   //重载=运算符  

    void Print();   //输出对象内容
};  

//默认构造函数的实现
NameStr::NameStr()
{
    static const char s_szDefaultName[]="Default name";
    static const char s_szDefaultStr[]="Default string";  

    m_pData=new char[strlen(s_szDefaultStr)+1];     //不能为为分配内存空间的字符指针赋值
    m_pName=new char[strlen(s_szDefaultName)+1];  

    strcpy(m_pName,s_szDefaultName);        //更规范的方式是使用strncpy函数进行拷贝
    m_pName[strlen(s_szDefaultName)]=‘\0‘;
    strcpy(m_pData,s_szDefaultStr);
    m_pData[strlen(s_szDefaultStr)]=‘\0‘;
}  

//析构函数的实现
NameStr::~NameStr()
{
    delete []m_pData;
    delete []m_pName;
}  

//带参构造函数的实现
NameStr::NameStr(const char* pName,const char* pData)
{
    m_pData=new char[strlen(pData)+1];      //开辟内存空间
    m_pName=new char[strlen(pName)+1];  

    strcpy(m_pData,pData);
    m_pData[strlen(pData)]=‘\0‘;
    strcpy(m_pName,pName);
    m_pName[strlen(pName)]=‘\0‘;
}  

//拷贝构造函数的实现
NameStr::NameStr(const NameStr& temp)
{
    m_pData=new char[strlen(temp.m_pData)+1];
    m_pName=new char[strlen(temp.m_pName)+1];  

    strcpy(m_pData,temp.m_pData);
    m_pData[strlen(temp.m_pData)]=‘\0‘;
    strcpy(m_pName,temp.m_pName);
    m_pName[strlen(temp.m_pName)]=‘\0‘;
}  

//重载=运算符的实现
NameStr& NameStr::operator=(const NameStr& temp)
{
    //首先要进行检查,防止自身复制
    if(&temp==this) //this是一个指针,表示本对象的地址。&temp是temp对象的指针。
    {
        return *this;
    }  

    //释放原有的内存空间
    delete []m_pData;
    delete []m_pName;  

    //分配新的内存空间
    m_pData=new char[strlen(temp.m_pData)+1];
    m_pName=new char[strlen(temp.m_pName)+1];  

    //进行拷贝
    strcpy(m_pData,temp.m_pData);
    m_pData[strlen(temp.m_pData)]=‘\0‘;
    strcpy(m_pName,temp.m_pName);
    m_pName[strlen(temp.m_pName)]=‘\0‘;  

    //返回本对象的引用
    return *this;
}  

inline void NameStr::Print()
{
    cout<<"Name:"<<m_pName<<endl;
    cout<<"String:"<<m_pData<<endl;
}  

//程序入口
int _tmain(int argc, _TCHAR* argv[])
{
    NameStr* pDefNss=NULL;  

    pDefNss=new NameStr[3];
    NameStr ns("hello","world");  

    delete []pDefNss;  

    NameStr ns1=ns;  

    return 0;
}  

原文地址:https://www.cnblogs.com/WHUT-Simon/p/11736217.html

时间: 2024-11-05 20:35:08

深拷贝与浅拷贝,还有一道比较好的面试题的相关文章

【转载】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 ()     {        cout<

完全理解python深拷贝和浅拷贝

import copya = [1, 2, 3, 4, ['a', 'b']]  #原始对象b = a  #赋值,传对象的引用c = copy.copy(a)  #对象拷贝,浅拷贝d = copy.deepcopy(a)  #对象拷贝,深拷贝a.append(5)  #修改对象aa[4].append('c')  #修改对象a中的['a', 'b']数组对象print 'a = ', aprint 'b = ', bprint 'c = ', cprint 'd = ', d 输出结果:a = 

javaScript之深拷贝与浅拷贝

js中有两种数据类型: 1.  基本类型  :  Number.String.Boolean.Null.Undefined 2.  复杂类型  :  Object .Array 深拷贝和浅拷贝只针对复杂类型的数据,因为基本类型数据的定义都会重新开辟新的内存. 浅拷贝拷贝的是内存地址,只是增加一个指针指向已有的内存,这时多个数据共用一个内存空间:深拷贝是新增了指针并且新开辟了内存空间,新指针指向新的内存. 浅拷贝: var a={ name:'WangJing', sex:'女', age:'25

C++拷贝构造函数(深拷贝,浅拷贝)

http://www.cnblogs.com/BlueTzar/articles/1223313.html 对于普通类型的对象来说,它们之间的复制是很简单的,例如:int a=88;int b=a; 而类对象与普通对象不同,类对象内部结构一般较为复杂,存在各种成员变量.下面看一个类对象拷贝的简单例子. 1 #include <iostream> 2 using namespace std; 3 4 class CExample { 5 private: 6 int a; 7 public: 8

python 中 深拷贝和浅拷贝的理解

在总结 python 对象和引用的时候,想到其实 对于python的深拷贝和浅拷贝也可以很好对其的进行理解. 在python中,对象的赋值的其实就是对象的引用.也就是说,当创建一个对象,然后赋给另外一个变量之后,实际上只是拷贝了这个对象的引用. 我们先用  利用切片操作和工厂方法list方法 来阐述一下浅拷贝. 举个栗子: Tom = ['Tom', ['age', 10]] Jack = Tom[:] ……切片操作 June = list(Tom) 接下来查看一下 上述三个变量的引用: >>

实现CMyString类--深拷贝与浅拷贝

实现CMyString类,实现赋值运算符函数(深拷贝与浅拷贝(写时拷贝)) 深拷贝:赋值运算符函数说明 步骤1.释放原来的内存空间 步骤2.再重新开辟要赋值的对象的大小的空间 步骤3.再将另一个对象的值拷贝给this对象 友元函数的说明:由于输出重载需要俩个参数,不能再隐含this指针,故使用友元函数 代码如下: <span style="font-size:18px;">class CMyString { public: CMyString() :str(new char

python 深拷贝和浅拷贝之可变和不可变对象总结

了解深拷贝和浅拷贝之前先要理解可变与不可变对象 python只允许使用引用传递,有可变对象和不可变对象,可变对象:list,dict.不可变对象有:int,string,float,tuple Python int,string,float,tuple不可变举栗子: def int_object(): i = 89 j = 89 print(id(89)) print('i id:' + str(id(i))) print('j id:' + str(id(j))) print(i is j)

深拷贝与浅拷贝

一.浅拷贝和深拷贝 所谓浅拷贝,就是由默认的拷贝构造函数所实现的对数据成员逐一赋值.若类中含有指针类型的数据,这种方式只是简单的把指针的指向赋值给新成员,但并没有给新成员分配内存,因此这种方式必然会导致错误.为了解决浅拷贝出现的错误,必须显示的定义一个拷贝构造函数,使之不但复制数据成员,而且为对象分配各自的内存空间,这就是所谓的深拷贝. 二.浅拷贝 浅拷贝就是由默认的拷贝构造函数所实现的数据成员逐一赋值.通常默认的拷贝构造函数能够胜任这个工作,但是若类中含有指针类型的数据,这种数据成员逐一赋值的

【转】 C++的深拷贝与浅拷贝

对于普通类型的对象来说,它们之间的复制是很简单的,例如:int a=88;int b=a; 而类对象与普通对象不同,类对象内部结构一般较为复杂,存在各种成员变量.下面看一个类对象拷贝的简单例子. 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 #include <iostream> using namespace std; class CExample { private:      int a; public:      CE