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

对象的动态创建和销毁以及对象的复制,赋值

🐶博主主页:@ᰔᩚ. 一怀明月ꦿ 

❤️‍🔥专栏系列:线性代数,C初学者入门训练,题解C,C的使用文章,「初学」C++​​​​​​​

🔥座右铭:“不要等到什么都没有了,才下定决心去做”

🚀🚀🚀大家觉不错的话,就恳求大家点点关注,点点小爱心,指点指点🚀🚀🚀

目录

🐰对象的动态创建和销毁

🐰对象的复制

🐰对象的赋值


🐰对象的动态创建和销毁

new和delete这两个运算符实现对内存的动态申请与释放的。如果要动态创建和销毁对象也使用这两个运算符

例如有一个类为Box,可以动态创建一个对象:

new Box;

这样系统就会从内存堆分配中一块内存空间,存放Box的对象,调用构造函数初始化对象。如果分配成功,new运算符会返回分配的内存的首地址;如果返回失败,则会返回一个NULL。但是通过new创建的对象没有名字,所以在使用new创建对象时都要声明一个指针变量保存对象的首地址,例如:

Box * ptr=new Box;

另外,还可以使用new创建对象时给出实参,调用有参的构造函数初始化对象

Box* ptr=new Box(2,2,2);

ptr指针就可以访问公用的成员了

ptr->volume();

如果动态创建对象失败,则会返回空指针,所以为了安全起见,可以判断指针是否为空

if(ptr!=NULL)

ptr->volume();

不需要使用动态创建的对象时,可以使用delete运算符销毁该对象

delete 指针名

delete ptr

这样就可以销毁ptr所指向的对象,将对象占用的空间归还给堆,使用new动态创建的对象只能通过delete进行销毁,系统不会进行自动销毁。如果不销毁,堆内存将被逐渐消耗。指针一旦指向动态创建的对象,就不要改变指针变量的值了,可能会造成动态创建创建的对象无法被销毁,也可能指针指向其他对象,delete销毁对象时可能会删错对象

🐰对象的复制

对象的复制是指在创建对象时使用已有对象快速复制出完全相同的对象
类名 对象2(对象1);    代入法
类名 对象2=对象1;    赋值法
其中对象1是和对象同类的并且已经存在的对象, 在这种情况下,系统会调用一个称为 “复制构造函数” 的特殊的构造函数。复制构造函数会将对象1的各数据成员的值逐个复制到对象2中相应的数据成员。 复制构造函数只有一个形参,这个形参就是本类的常引用。 复制构造函数的函数体主要是将形参中对象的各数据成员值赋给自己的数据成员,为保证数据安全,引用加上const,看看复制构造函数的函数形式:​​​​​​​
Box::Box(const Box &c)
{
    length=c.length; width=c.width; height=c.height;
}
即使程序中没有定义复制构造函数(每一个类都有一个复制构造函数),编译器会隐式地提供一个。即使定义了其他的构造函数,编译器也会提供一个复制构造函数,他会将实参对象的非static数据成员逐个复制到创建的对象中。
普通的构造函数和复制构造函数有哪些区别呢
(1)在形式上普通构造函数一般是形参列表,创建对象时通过实参列表给出初始化对象所需要各个数据成员的值。而复制构造函数的形参则只有一个,及本类对象的引用。
(2)在调用时系统会根据实际参数列表的类型来自动选择调用
(3)调用情况不一样,普通构造函数是创建对象时由系统自动调用;而复制构造函数是在使用已有的对象复制一个新对象时系统自动调用。以下三种情况才会复制对象:(1)创建一个新对象,并用同类对象初始化它。(2)函数参数是类对象(3)函数返回值是类对象
如果数据成员有指针变量时,复制构造函数会出现指针悬挂问题。
#include<iostream>
using namespace std;
class person
{
public:
    person(char* Name,int Age);
    person(const person &temp);
    ~person();
    void setAge(int x)
    {
        age=x;
    }
    void print();
private:
    char* name;
    int age;
};
person:: person(const person &temp)//复制构造函数
{
    name=new char[strlen(temp.name)+1];
    strcpy(name,temp.name);
    age=temp.age;
    cout<<"persson is called!!!"<<endl;
}
person::person(char* Name,int Age)//构造函数
{
    name=new char[strlen(Name)+1];
    strcpy(name,Name);
    age=Age;
    cout<<"persson is called!!!"<<endl;
}
person::~person()
{
    cout<<"persson is called"<<endl;
    delete []name;
    name=NULL;
}
void person::print()
{
    cout<<"name:  "<<name<<"age:  "<<age<<endl;
}
int main()
{
    person s1("张三",23);
    person s2(s1);
    s1.setAge(1);
    s2.setAge(2);
    s1.print();
    s2.print();
    return 0;
}
因此需要我们自己定义一个复制构造函数

🐰对象的赋值

对象赋值的一般形式:
对象1=对象2;

🌸🌸🌸如果大家还有不懂或者建议都可以发在评论区,我们共同探讨,共同学习,共同进步。谢谢大家! 🌸🌸🌸    


http://www.kler.cn/a/1249.html

相关文章:

  • Linux之Kobject
  • CSS鼠标悬浮及其样式
  • Spring 设计模式:经典设计模式
  • Springboot——钉钉(站内)实现登录第三方应用
  • sql server cdc漏扫数据
  • 人工智能与物联网:智慧城市的未来
  • 深入剖析Linux——进程信号
  • SpringCloud五大核心组件
  • Python每日一练(20230318)
  • 深入理解 Go slice 扩容机制
  • Redis基础篇
  • Spring 事务(编程式事务、声明式事务@Transactional、事务隔离级别、事务传播机制)
  • Spring事务和事务传播机制
  • 插件化架构设计(2):插件化从设计到实践该考量的问题汇总
  • 菱形继承和C++相关问题
  • React 用一个简单案例体验一遍 React-dom React-router React-redux 全家桶
  • Springboot集成Swagger
  • 公司测试员用例写得乱七八糟,测试总监制定了这份《测试用例编写规范》
  • 【高阶数据结构】红黑树
  • css属性学习
  • Java基础之常见运算符
  • 77.qt qml-QianWindow-V1版本界面讲解
  • @JsonFormat与@DateTimeFormat
  • go语言如何使用new构造Map
  • 【技术方案】常见库存设计方案-各种方案对比总有一个适合你
  • 百度的文心一言 ,没有想像中那么差