解锁即可观看《物联网/嵌入式工程师》完整课程视频

物联网/嵌入式工程师

重磅升级:新增硬件电路设计与实战,让你软硬通吃,同级PK无敌!行业风口、政策倾斜,新晋热门高薪不内卷!0基础一站式就业完整路径,抢占先发优势!

【第1周】迈进供不应求的物联网/嵌入式行业大门
【第2周】C语言进阶-编程思想
【第3周】C语言高级进阶
【第4周】C语言项目实战与《阶段笔试》
【第5周】数据结构-线性结构、非线性结构等
【第6周】排序与复杂度&数据结构项目实战
【第7周】Shell编程和Makefile工程管理
【第8周】Linux文件IO和标准IO
【第9周】Linux下文件操作项目实践与阶段评测
【第10周】Linux 多进程、多线程、IO模型
【第11周】Linux 项目实战-实现并发服务器模型与企业笔试
【第12周】网络基础和UDP Socket编程
【第13周】TCP Socket编程和WireShark抓包分析
【第14周】网络编程项目实战-网络视频监控与企业笔试
【第15周】从C到C++
【第16周】软件设计模式与C++11新特性
【第17周】项目实战-C++语言实现五子棋游戏与企业笔试
【第18周】嵌入式产品人机交互必备-QT框架
【第19周】项目实战-QT开发音乐播放器
【第20周】智能硬件开发-ARM核介绍和基础外设
【第21周】智能硬件开发-单片机常用外设
【第22周】stm32芯片-智能硬件项目实战与企业笔试
【第23周】大厂必备- linux内核与文件系统移植
【第24周】嵌入式开发-系统移植-bootloader、yocto
【第25周】嵌入式底层核心技能-Linux设备驱动初级
【第26周】嵌入式底层核心技能-Linux设备驱动中级
【第27周】嵌入式底层核心技能-Linux设备驱动高级1
【第28周】嵌入式底层核心技能-Linux设备驱动高级2
【第29周】智能家居项目实战之Linux智能网关端开发
【第30周】智能家居项目实战之STM32单片机设备端开发
【第31周】人脸指纹识别考勤机设计与实战
【第32周】硬件电路设计-电路基础知识
【第33周】STM32最小系统 - 硬件电路设计实战项目(一)
【第34周】CAN总线分析仪 - 硬件电路设计实战项目(二)
【第35周】4路输入输出控制器 - 硬件电路设计实战项目(三)
【第36周】8路逻辑分析仪 - 硬件电路设计实战项目(四)
【第37周】项目答辩和就业指导&独立开发阶段-三大热门领域项目
章节
问答
课签
笔记
评论
占位
占位

复制一个员工:赋值构造函数

上一小节中,我们介绍了构造函数和析构函数。这一小节,我们来介绍一个特殊的构造函数。

先来看一个例程:

int main(int argc,char **argv)
{
    Staff staffA;
    Staff staffB = staffA;

    return 0;
}

我们先实例化了一个对象 staffA,然后又实例化了一个对象 staffB。希望 staffB 和 staffA 具有相同的内容,或者说希望 staffB 是 staffA 的副本,那么我们一般直接在实例化的时候进行赋值就可以了。

Staff staffB = staffA;

这样做了之后,C++ 会自动为我们拷贝 staffA 中的成员变量到 staffB 的成员变量中,但是这种自动拷贝机制在某些情况下无法完成我们想要的动作,甚至有可能会出错。我们来具体看一下。

在 Staff 类中添加一些内容:

Staff.hpp

#include <string>

class Staff
{
public:
    Staff(std::string _name, int _age);
    ~Staff();

public:
    std::string name;
    int age;

    char * mem = nullptr;
};

Staff.cpp

#include "Staff.hpp"
#include <stdio.h>

Staff::Staff(std::string _name, int _age)
{
    mem = (char *)malloc(20);
    name = _name;
    age = _age;
    printf("构造函数被调用\n");
}

Staff::~Staff()
{
    if(mem != nullptr){
        free(mem);
        mem = nullptr;
    }
    printf("析构函数被调用\n");
}

在上面的代码中,在类中定义了一个指针,在构造函数中,通过 malloc 函数分配了一个 20 字节大小的堆内存,然后将这片堆内存的首地址赋值给了这个指针。在析构函数中,我们将这片堆内存释放掉。

这个时候,我们再进行一开始的操作:

Staff staffB = staffA;

先来看看 staffA 在实例化之后的内存布局:

mem 指针指向了一片 20 个字节大小的堆内存。

这个时候,再来看看执行了Staff staffB = staffA;之后,staffB 的内存布局会怎么样:

可以看到,在 C++ 默认的复制模式之下,两个对象中的 mem 指针指向了同一片内存。因为 C++ 默认的复制模式只会简单得把成员的值进行复制,面对这个 mem 指针,他只会把指针的值进行拷贝,最后的结果就是 mem 指针指向了同一片内存。

这种拷贝方式,被称之为浅拷贝。

赋值构造函数

Staff staffB = staffA;

当我们使用这种方式实例化对象的时候,并不会调用普通构造函数,而是会调用一个特殊的构造函数,被称之为赋值构造函数或者拷贝构造函数。如果我们没有写,那么就会按照浅拷贝的方式来进行复制。一个拷贝构造函数看起来就像下面这样:

Staff(const Staff & staff);

这个函数中只有一个参数 staff,表示要拷贝的对象,在我们的例子中,就是 staffA。(const 和 & 我们在后续的课程中会具体讲解)

那么我们来完整的编写一下这个函数

Staff.hpp

#include <string>

class Staff
{
public:
    Staff(std::string _name, int _age);
    Staff(const Staff & staff);
    ~Staff();

public:
    std::string name;
    int age;

    char * mem = nullptr;
};

Staff.cpp

#include "Staff.hpp"
#include <stdio.h>

Staff::Staff(std::string _name, int _age)
{
    mem = (char *)malloc(20);
    name = _name;
    age = _age;
    printf("构造函数被调用\n");
}

Staff::Staff(const Staff & staff)
{
    name = staff.name;
    age = staff.age;

    mem = (char *)malloc(20);
    memcpy(mem, staff.mem, 20);
}

Staff::~Staff()
{
    if(mem != nullptr){
        free(mem);
        mem = nullptr;
    }
    printf("析构函数被调用\n");
}

任务

?不会了怎么办
||

提问题

写笔记

公开笔记
提交
||

请验证,完成请求

由于请求次数过多,请先验证,完成再次请求

加群二维码

打开微信扫码自动绑定

您还未绑定服务号

绑定后可得到

  • · 粉丝专属优惠福利
  • · 大咖直播交流干货
  • · 课程更新,问题答复提醒
  • · 账号支付安全提醒

收藏课程后,能更快找到我哦~

使用 Ctrl+D 可将课程添加到书签

邀请您关注公众号
关注后,及时获悉本课程动态

举报

0/150
提交
取消
全部 精华 我要发布
全部 我要发布
最热 最新
只看我的

手记推荐

更多

本次提问将花费2个积分

你的积分不足,无法发表

为什么扣积分?

本次提问将花费2个积分

继续发表请点击 "确定"

为什么扣积分?