当前位置: 首页 > news >正文

【直击招聘C++】2.6 对象之间的复制

2.6 对象之间的复制

  • 一、要点归纳
    • 1. 对象之间的复制操作
      • 1.1 =运算符
      • 1.2 拷贝构造函数
    • 2. 对象之间的浅复制和深复制
      • 2.1 对象的浅复制
      • 2.2 对象的深复制
  • 二、面试真题解析
    • 面试题1
    • 面试题2

一、要点归纳

1. 对象之间的复制操作

  同一个类的对象之间可以进行复制操作,即将一个对象数据成员复制给另一个对象的相应的数据成员。

1.1 =运算符

  当已经定义了同一个类的多个对象时,可以在这些对象之间进行赋值运算,通常采用=运算符来实现,例如:

MyClass s1,s2;
s1=s2;

  上述对象复制语句的功能只是将s2的所有数据成员赋值给s1相应的数据成员。如果程序员在类中没有定义=运算符,编译器会相建立默认构造函数那样建立一个默认的=运算符。实际上,=运算符对应operator=成员函数,所谓定义=运算符就是重载operator=成员函数。

1.2 拷贝构造函数

  如果创建新对象并赋值,调用拷贝构造函数。在定义新对象并初始化时,形如A x=y的语句是通过调用拷贝构造函数创建对象x,通常将对象y的数据成员赋值给对象x相应的数据成员,例如有以下程序:

class A
{int m;public:A(){m = 0;std::cout << "默认构造函数" << std::endl;}~A(){std::cout << "析构函数" << std::endl;}A(int n){m = n;std::cout << "重载构造函数" << std::endl;}A(const A &b){std::cout << "拷贝构造函数" << std::endl;}A &operator=(const A &b){std::cout << "operator=" << std::endl;return *this;}
};void test01()
{A a(2), b(a), c;c = a;A d = a;
}

输出结果如下:

重载构造函数
拷贝构造函数
默认构造函数
operator=
拷贝构造函数
析构函数
析构函数
析构函数
析构函数

  建立A a(2)调用重载构造函数,执行b(a)调用拷贝构造函数,执行A c调用默认构造函数。执行c=a调用operator=成员函数,执行A d=a调用拷贝构造函数。依次销毁d\c\b\a调用析构函数。

2. 对象之间的浅复制和深复制

  无论是拷贝构造函数还是=运算符,都可春实现对象复制。对象复制又分为浅复制和深复制。

2.1 对象的浅复制

  当两个对象之间进行复制时,若复制完成后她们还共享某些资源(内存空间),其中一个对象的销毁会影响另一个对象,这种对象之间的复制称为浅复制。例如有以下程序:

class Student
{int no;char *pname;public:Student() {}            // 默认构造函数Student(int n, char *p) // 重载构造函数{no = n;pname = new char[10]; // 用new分配内存空间strcpy(pname, p);}Student(Student &s) // 拷贝构造函数{no = s.no;pname = s.pname;}void display(){std::cout << "no:" << no << ",name:" << pname << std::endl;std::cout << "pname:" << (int *)pname << std::endl; // 输出pname地址}
};
void test02()
{Student s(10, (char *)"Mary"), t(s);std::cout << "s:";s.display();std::cout << "t:";t.display();
}

输出如下:

s:no:10,name:Mary
pname:0x55d1366c22c0
t:no:10,name:Mary
pname:0x55d1366c22c0

在上述程序中先声明了一个类Student,其中有一个拷贝构造函数。在main函数中建立了一个对象s,通过拷贝构造函数由s建立t对象。由Student类的拷贝构造函数可以看到s到t的复制是浅复制,如下图所示,因为这两个对象的pname数据成员均指向相同的内存空间。

2.2 对象的深复制

  当两个对象之间进行复制时,若复制完成后她们不会共享任何资源(内存空间),其中一个对象的销毁不会影响到另一个对象,这种对象之间的复制称为深复制。例如将上述程序修改如下:

class Student
{int no;char *pname;public:Student() {}            // 默认构造函数Student(int n, char *p) // 重载构造函数{no = n;pname = new char[10]; // 用new分配内存空间strcpy(pname, p);}Student(Student &s) // 拷贝构造函数{no = s.no;// pname = s.pname;pname = new char[strlen(s.pname + 1)];strcpy(pname, s.pname);}void display(){std::cout << "no:" << no << ",name:" << pname << std::endl;std::cout << "pname:" << (int *)pname << std::endl; // 输出pname地址}~Student() { delete pname; } // 析构函数
};void test03()
{Student s(10, (char *)"Mary"), t(s);std::cout << "s:";s.display();std::cout << "t:";t.display();
}
s:no:10,name:Mary
pname:0x55acd2a8b2c0
t:no:10,name:Mary
pname:0x55acd2a8b2e0

修改后类Student的亏被构造函数在复制数据成员pname时另外开辟内存空间,从而当执行Student s(10, (char *)“Mary”), t(s);的时候由对象s复制产生对象t时,这两队对象的pname成员分别指向不同的内存单元,如图?

二、面试真题解析

面试题1

【面试题】什么时候必须重写拷贝构造函数?
【答】当构造函数涉及动态存储分配空间时要自己写拷贝构造函数,并且要深复制。

面试题2

【面试题】不考虑任何编译器优化,下面代码的第8行会发生(一次拷贝构造、一次析构函数、一次operator=)

#include <iostream>
class B
{
};
B func(const B & rhs)
{return rhs;
}
void main()
{B b1,b2;b2=func(b1);
}

【答】首先调用一次拷贝构造函数将b1复制给形参rhs,返回rhs并赋值给b2调用一次operator=,func函数执行完毕销毁rhs调用一次析构函数。

http://www.lryc.cn/news/2710.html

相关文章:

  • 学了这么久python,不会连自己啥python版本都不知道吧?
  • Revive:从间谍软件进化成银行木马
  • Python 之 NumPy 简介和创建数组
  • 与六年测试工程师促膝长谈,他分享的这些让我对软件测试工作有了全新的认知~
  • 裕太微在科创板上市:市值约186亿元,哈勃科技和小米基金为股东
  • 毕业后5年,我终于变成了月薪13000的软件测试工程师
  • 实践指南|如何在 Jina 中使用 OpenTelemetry 进行应用程序的监控和跟踪
  • MySQL 创建数据表
  • 一文详解网络安全事件的防护与响应
  • vue directive 注册局部指令
  • LC-70-爬楼梯
  • Scratch少儿编程案例-可爱的简约贪吃蛇
  • 编译 Android 时如何指定输出目录?
  • CF1574C Slay the Dragon 题解
  • 创建Django项目
  • CUDA中的统一内存
  • 利用机器学习(mediapipe)进行人脸468点的3D坐标检测--视频实时检测
  • 事务基础知识与执行计划
  • 数据库实践LAB大纲 06 INDEX
  • 网络安全实验室6.解密关
  • 了解并发编程
  • (C语言)程序环境和预处理
  • RiProV2主题美化增加支付页底部提示语ritheme主题美化
  • 2022年文章分类整理
  • 蓝牙设备中的Device UUID 与 Service UUID
  • 【学习记录】PCA主成分分析 SVD奇异值分解
  • 用 Python 调用 GPT-3 API
  • 类和对象实操之【日期类】
  • 微搭中如何实现弹性布局
  • 九龙证券|外资强势出手!这只科创板百元股,被疯狂加仓