c++入门学习④——对象的初始化和清理

目录

对象的初始化和清理:

why?

如何进行初始化和清理呢?

使用构造函数和析构函数​编辑

构造函数语法:

析构函数语法:

构造函数的分类:

两种分类方式:

三种调用方法:

括号法(默认构造函数调用)

显示法

隐式转换法

拷贝调用函数的调用时机

1.使用一个已经常见完毕的对象来初始化一个新对象

2.值传递的方式给函数参数传值。

3.以值方式返回局部对象

构造函数调用时机:

深拷贝浅拷贝:

有可能的错误:

错误原因:

解决方案:

 深拷贝和浅拷贝案例:

 初始化列表

基本初始化列表案例:

灵活定义初始化的数字时的初始化列表案例:

类对象作为类成员

静态成员

静态成员变量:

 静态成员变量访问案例:

访问方式:

静态成员函数

静态成员函数创建调用基本示例

why?

结语:


本篇博客讲解c++入门学习④有关对象的初始化以及清理的知识点💪

传送门:

c++入门学习①-CSDN博客

c++入门学习②-CSDN博客

c++入门学习③——封装-CSDN博客

对象的初始化和清理:

why?

为了保证数据安全👉一个对象或者是变量没有初始化,则后面的后果是未知的,同样,没有清理也是可能会造成一系列的安全问题。

如何进行初始化和清理呢?

使用构造函数和析构函数
构造函数语法:

类名(){}
1.构造函数,没有返回值也不写void
2.函数名称与类名相同
3.构造函数可以有参数,因此可以发生重载
4.程序在调用对象时候会自动调用构造,无须手动调用,而且只会调用一次

自己不写,程序会自动帮你实现,不过函数里面的代码是空的,要是自己写,程序运行你自己写的构造函数

析构函数语法:

~类名(){}
1.析构函数,没有返回值也不写void
2.函数名称与类名相同,在名称前加上符号
3.析构函数不可以有参数,因此不可以发生重载
4.程序在对象销毁前会自动调用析构,无须手动调用,而且只会调用一次

自己不写,程序会自动帮你实现,不过函数里面的代码是空的,要是自己写,程序运行你自己写的析构函数

代码:

#include<iostream>
#include<string>
using namespace std;
class person{public://创造构造函数person() {cout << "构造函数" << endl;}~person() {cout << "析构函数调用" << endl;}
};
void test1() {person a;
}//局部变量,在栈区上创建,test执行完毕后,释放对象,会自动实现析构函数,不过是空的
//当自己创建了一个,那么就执行自己创建的析构函数int main()
{test1();system("pause");return 0;
}

构造函数的分类:

两种分类方式:

按照参数分类:有参数和无参数(默认情况下为无参)

按照类型分类:普通和拷贝

拷贝:copy、复制,

例如:person(const person &a){name=a.name;};把其他的属性拷贝到这个上面

拷贝方式创建,注意括号里写要copy的对象,并加上const,以引用的方式写入。

三种构造函数的具体表示:

#include<iostream>
#include<string>
using namespace std;
class person{public://创造构造函数person() {cout << "默认构造函数" << endl;}person(int age) {cout << "有参构造函数" << endl;Age = age;}person(const person& p) {cout << "拷贝构造函数" << endl;Age = p.Age;}~person() {cout << "析构函数调用" << endl;}int Age;
};

引用:c++引用-CSDN博客

三种调用方法:

括号法(默认构造函数调用)

person p1;调用无参构造函数,也是默认构造函数调用;

person p2(12);调用有参构造函数

person p3(p2);调用拷贝构造函数

注意事项:调用默认构造函数的时候不要加(),因为编译器会认为这个是一个函数声明,不会认为在创建对象

显示法

person p1;默认构造

person p2=person(12);有参构造

person p3=person(p2);拷贝构造

单独拿出来person(12),称为匿名对象,当前执行结束后,系统会立即回收掉匿名对象,马上析构掉

不单独看这个,匿名对象就有名字了,左边的就是它的名字

注意:不要利用拷贝构造函数初始化匿名对象。编译器会认为person(p3)==person p3;认为这是一个对象声明

隐式转换法

person p4=10;相当于写了person p4=person(10)有参构造

person p5=p4;拷贝构造

括号法最简单,这里推荐使用这个


拷贝调用函数的调用时机

1.使用一个已经常见完毕的对象来初始化一个新对象

#include<iostream>
using namespace std;
class person{public://创造构造函数person() {cout << "默认构造函数" << endl;}person(int age) {Age = age;cout << "有参构造函数" << endl;}person(const person & a) {Age = a.Age;cout << "拷贝构造函数" << endl;}~person() {cout << "析构函数调用" << endl;}int Age;
};
void test1() {person a;//局部变量,在栈区上创建,test执行完毕后,释放对象
}
//使用一个已经创建完毕的对象来初始化一个新的对象
void test2() {person p(20);//调用有参构造函数person p1(p);//调用拷贝构造函数
}int main()
{//test1();test2();system("pause");return 0;
}

输出结果: 

2.值传递的方式给函数参数传值。

#define  _CRT_SECURE_NO_WARNINGS
#include<iostream>
#include<string>
using namespace std;
class person{public://创造构造函数person() {cout << "默认构造函数" << endl;}person(int age) {Age = age;cout << "有参构造函数" << endl;}person(const person & a) {Age = a.Age;cout << "拷贝构造函数" << endl;}~person() {cout << "析构函数调用" << endl;}int Age;
};
//值传递的方式给函数参数传值
void test(person p) 
{
}
void test3() {person p;//调用默认构造函数test(p);//调用拷贝构造函数,因为这里是以值传入,相当于拷贝一个副本传进去}int main()
{//test1();//test2();test3();return 0;
}

3.以值方式返回局部对象

示例:

#define  _CRT_SECURE_NO_WARNINGS
#include<iostream>
#include<string>
using namespace std;
class person{public://创造构造函数person() {cout << "默认构造函数" << endl;}person(int age) {Age = age;cout << "有参构造函数" << endl;}person(const person & a) {Age = a.Age;cout << "拷贝构造函数" << endl;}~person() {cout << "析构函数调用" << endl;}int Age;
};//以值返回局部对象
person test0() {person p;//局部对象在完成函数之后会被自动释放掉,但是现在以值方式返回,则在返回的时候会自动给创建一个新的对象cout << "p地址" << (int*) & p << endl;return p;//返回的是p拷贝的一个新的对象
}
void test4() {person a=test0();//这里的函数a没有调用析构函数和构造函数cout << "a地址" << (int*)&a << endl;
}
int main()
{//test1();//test2();//test3();test4();system("pause");return 0;
}

构造函数调用时机:

默认情况添加的函数:

⭐注意:拷贝构造函数默认情况下只写值传递语句,不会出现输出语句

构造函数调用规则如下:
若用户定义有参构造函数,c++不再提供默认无参构造,但是会提供默认拷贝构造函数

当仅定义有参构造函数时不要进行默认无参构造,会发生错误显示没有默认无参构造函数。

若用户定义拷贝构造函数,c++不会再次提供其他构造函数

只定义拷贝构造函数时如果使用其他的构造函数,则会发生错误

深拷贝浅拷贝:

一般浅拷贝是逐个字节依次复制过去,而默认情况下系统创建拷贝构造函数是浅拷贝

有可能的错误:

一般使用不会出错,但是如果数据是程序员开发在堆区,就会发生释放重复的错误

错误原因:

因为浅拷贝使两个对象里的属性指向同一块空间,当释放一个对象中的属性之后,再去释放另一个也在堆区的数据,由于她们指向相同,所指向的空间已经被释放,但是系统不知道因此会发生错误

解决方案:

改进方法就是进行深拷贝,程序员自己创建一个函数来达到深拷贝的效果,深拷贝是在拷贝构造函数内进行自己再开辟一片空间,只复制要拷贝对象属性的内容而非空间地址。

 深拷贝和浅拷贝案例:

#include<iostream>using namespace std;
class person{public://创造构造函数person() {cout << "默认构造函数" << endl;}person(int age,int height) {Age = age;high = new int(height);cout << "有参构造函数" << endl;}person(const person & a) {Age = a.Age;cout << "person的拷贝构造函数" << endl;high=new int(*a.high);}~person() {//析构代码,将堆区开辟数据做释放操作if (high != NULL) {delete high;high = NULL;}//加上这个代码后运行,发现程序崩溃了?cout << "析构函数调用" << endl;}int Age;int* high;//身高,使用指针,把这个参数存放在堆区//堆区的数据,由程序员手动开辟,也由程序员手动释放,销毁前释放,在析构之前释放
};
void test1() {person p1(18,160);cout << "p1年龄:" << p1.Age<<"p1身高为:"<<*p1.high << endl;person p2(p1);//编译器创建了一个拷贝函数,进行了一个浅拷贝的操作,简单的赋值操作,逐字节拷贝cout << "p2年龄:" << p2.Age<< "p2身高为:" <<*p2.high << endl;//栈区先进后出,所以p2先被释放,p2先执行析构函数,p2释放过了指针所指的内存,则再次用p1释放时发生错误//解决:浅拷贝用深拷贝改进,不是逐个字节拷贝了,是重新开辟一个空间,两个空间就不会存在重复释放了//自己写拷贝构造函数
}
int main()
{test1();system("pause");return 0;
}

 初始化列表

简介:
初始化类中属性:1创建构造函数2.初始化列表
构造函数():属性1(值1),属性2(值2)

基本初始化列表案例:

#include<iostream>
using namespace std; //初始化列表//初始化类中属性:1创建构造函数2.初始化列表// 构造函数():属性1(值1),属性2(值2)
class person{public://初始化列表给类属性赋初值;person() :Age(10),b(20){}int Age;int b;
};
void test1() {person p1;cout << p1.Age <<"\t" << p1.b << endl;
}
int main()
{test1();system("pause");return 0;
}

灵活定义初始化的数字时的初始化列表案例:

#include<iostream>using namespace std; //初始化列表//初始化类中属性:1创建构造函数2.初始化列表// 构造函数():属性1(值1),属性2(值2)
class person{public://初始化列表给类属性赋初值;person(int a,int n) :Age(a),b(n)//注意这个冒号的位置,别写错了{}int Age;int b;
};
void test1() {person p1(12,22);//12先传给了p1中的int a,然后传入到Age里的a,这样Age就被这个数字赋值了,被初始化了。cout << p1.Age <<"\t" << p1.b << endl;
}
int main()
{test1();system("pause");return 0;
}

类对象作为类成员

类定义的对象称为另一个类里的成员,则称为类成员

构造顺序和析构顺序相反

构造时,先构造类成员的类,再构造类成员所在的类

而析构顺序则正相反。

#include<iostream>
#include<string>
using namespace std; 
//类对象成为另一个对象的成员,称为类成员
//先有的什么?👉先构造其他的类的对象,再构造这个类
//那么析构的顺序呢?和构造的顺序是相反的。
class A {
public:// 可以用Phone a_phone=pname👉隐式转换法来初始化变量A(string Name, string pname):name(Name),a_phone(pname){cout << "A的构造函数" << endl;}//姓名string name;//手机Phone a_phone;//类成员
};
//手机类
class Phone{
public:Phone(string pname) {//Phone  的构造函数Pname = pname;cout << "Phone的构造函数" << endl;}//手机品牌名string Pname;
};
void test1() {A p("黑菜钟","华为");cout << "名字:"<<p.name<< endl;cout <<"手机型号:"<< p.a_phone  << endl;}
int main()
{test1();system("pause");return 0;
}

静态成员

定义:静态成员就是在成员变量和成员函数前加上关键字static,称为静态成员

分为:静态成员变量和静态成员函数

静态成员变量要求:所有对象共享同一份数据,编译阶段分配内存,类内声明,类外初始化

静态成员函数要求:

静态成员变量:

静态成员变量示例:

当静态成员变量像平常一样定义类对象并且输出所对应的值的时候,程序会怎么样?

结果:发生错误,无法解析的外部命令👉这是因为静态成员变量要类外进行初始化操作,这个程序显然没做

如何在类外初始化呢?👉int person::m = 100;(这里的person是类名,m是类内静态变量)

 静态成员变量访问案例:

⭐静态成员变量不属于某个对象,所有对象共享同一份

访问方式:

1.通过对象访问
person p;
cout<<p.m_A<<endl;
2.通过类名访问
cout << person::m_A << endl;

#include<iostream>
#include<string>
using namespace std; 
//静态成员变量
//所有对象共享同一份数据
// 编译阶段分配内存
// 类内声明,类外初始化
class person {
public:static int m_A;};
int person::m_A = 10;
void test1() {person p;cout << p.m_A << endl;//10person p2;p2.m_A = 20;cout << p.m_A << endl;//结果为20,因为所有对象共享数据,一改全改//静态成员变量不属于某个对象,所有对象共享同一份
}
void test2() {//访问方式1.通过对象访问person p;cout<<p.m_A<<endl;//20//访问方式2.通过类名访问cout << person::m_A << endl;//用::来借助访问类对象
}int main()
{test1();test2();system("pause");return 0;
}

注意:静态成员变量只能在公共权限内定义,在私有权限内定义不出错,但是不能共享数据。类外访问不到私有静态变量。

静态成员函数

特点:所有对象共享同一个函数,静态成员函数只能访问静态成员变量

静态成员函数创建调用基本示例

#include<iostream>
#include<string>
using namespace std; 
//静态成员函数
//所有对象共享同一个函数
// 静态成员函数只能访问静态成员变量class person {
public:static void func() {cout << "static void func调用" << endl;}
};void test1() 
{//访问方式和静态成员变量的访问方式相同//1.通过对象person p;p.func();//2.通过类名person::func();
}int main()
{test1();system("pause");return 0;
}

如果静态成员函数访问非静态成员变量会出现👉这样的结果

why?

数据属于特定的,而函数体无法区分是哪个对象的,会出错

而静态成员变量是共享的,所有类的对象都用一份,函数体不需要知道这是哪个类对象对应的

⭐注意: 类外还是不可以访问私有的静态成员函数

结语:

希望这篇有关c++对象初始化和清理的博客对大家有所帮助,欢迎大佬们留言o(* ̄▽ ̄*)ブ

一起学习进步!!!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/671107.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

UE中对象创建方法示例和类的理解

对象创建方法示例集 创建Actor示例 //创建一个护甲道具 AProp* armor GetWorld()->SpawnActor<AProp>(pos, rotator); 创建Component示例 UCapsuleComponent* CapsuleComponent CreateDefaultSubobject<UCapsuleComponent>(TEXT("CapsuleComponent&qu…

【C#】创建Json文件并根据dll路径获取

创建Json文件 更改属性 【代码】根据dll路径获取 Assembly assembly Assembly.GetExecutingAssembly(); string assemblyPath assembly.Location; string relativeDllPath System.IO.Path.Combine(System.IO.Path.GetDirectoryName(assemblyPath), "Json\\test.json&q…

liunx服务器下vim无法:wq解决方案

当你vim某一个文件是&#xff0c;由于其他人在操作&#xff0c;或者上次操作中断&#xff0c;造成的 解决方案&#xff0c;不要&#xff0c;vim -r /etc/****&#xff0c;不然你会得到 正确方案 然后在&#xff0c;直接sudo vim /ect/xxx&#xff0c;还出现第一幅图就直接选…

LabVIEW高精度微小电容测量

LabVIEW高精度微小电容测量 在电子工程和科研领域&#xff0c;精确测量微小电容值是一项有一定要求的任务&#xff0c;尤其在涉及到高精度和低成本时。设计了一种基于LabVIEW高精度微小电容测量系统&#xff0c;旨在提供一个既经济又高效的解决方案。 该系统的核心在于使用FD…

陪诊系统|陪诊小程序|陪诊服务让就医更容易

陪诊系统已经出现了好几年。尤其是这两年&#xff0c;它得到了人们的广泛认可。陪诊行业的快速发展主要是因为人们对这个行业的需求非常大。目前&#xff0c;我国面临着严重的老龄化问题&#xff0c;生活节奏也越来越快&#xff0c;有时候无法亲自陪伴在老人的身边。陪诊工作人…

openGauss学习笔记-215 openGauss性能调优-确定性能调优范围-性能日志

文章目录 openGauss学习笔记-215 openGauss性能调优-确定性能调优范围-性能日志215.1 性能日志概述215.2 性能日志收集的配置参数 openGauss学习笔记-215 openGauss性能调优-确定性能调优范围-性能日志 215.1 性能日志概述 性能日志主要关注外部资源的访问性能问题。 性能日…

记录一次centos 使用selenium运行环境

这里写自定义目录标题 宝塔面板 安装 selenium安装google-chrome 宝塔面板 安装 selenium 安装google-chrome yum install https://dl.google.com/linux/direct/google-chrome-stable_current_x86_64.rpm 查看chrome版本 google-chrome --version 下载对应chrome版本的chro…

VScode上无法运行TSC命令,Typescript

如何解决问题 第一步&#xff1a;使用 winx 快捷键&#xff0c;会出现如下弹窗&#xff0c;鼠标左键单击Windows PowerShell 即可打开shell 第二步&#xff1a;运行 set-ExecutionPolicy RemoteSigned 命令&#xff0c;在询问更改执行策略的时候选择敲Y或者A 第三步&#xff…

【C语言初阶-结构体】关于结构体的声明定义、结构体传参详解

目录 1. 结构体的声明 1.1 结构的基础知识 1.2 结构的声明 1.3 结构成员的类型 1.4 结构体变量的定义和初始化 2. 结构体成员的访问 2.1(.)操作符 2.2&#xff08;->&#xff09;操作符 3.结构体传参 1. 结构体的声明 1.1 结构的基础知识 结构体是一些值的集合&…

uniapp vue3怎么调用uni-popup组件的this.$refs.message.open() ?

vue2代码 <!-- 提示信息弹窗 --><uni-popup ref"message" type"message"><uni-popup-message :type"msgType" :message"messageText" :duration"2000"></uni-popup-message></uni-popup>typ…

算法笔记刷题日记——3.简单入门模拟 3.2 查找元素

刷题日记 3.2 查找元素 B1041 B1004 B1028 B1032 A1011 A1006 A1036 错题记录 B1028 人口普查 某城镇进行人口普查&#xff0c;得到了全体居民的生日。现请你写个程序&#xff0c;找出镇上最年长和最年轻的人。 这里确保每个输入的日期都是合法的&#xff0c;但不一定是合理的…

开关电源学习之Boost电路

如果我们需要给一个输入电压为5V的芯片供电&#xff0c;而我们只有一个3.3V的电源&#xff0c;那怎么办&#xff1f; 我们能不能把3.3V的电压升到5V&#xff1f; 一、电感的简介 而在升压的电路设计方案中&#xff0c;使用到一个重要的元器件&#xff1a;电感。 电感的特性…

【PyQt】05-多线程

文章目录 前言一、什么是单线程、多线程二、代码现象示例多线程代码运行结果 总结 前言 文章开始还是解释一下&#xff0c;这是跟着王铭东老师学习的。 提示&#xff1a;以下是本篇文章正文内容&#xff0c;下面案例可供参考 一、什么是单线程、多线程 单线程 在Python中&am…

PDF文件格式(一):交叉引用流

在PDF-1.5版本之前&#xff0c;对象的交叉引用信息是存储在交叉引用表(cross-reference table)中的。在PDF-1.5版本之后&#xff0c;引进了交叉引用流(cross-reference stream)对象&#xff0c;可以用它来存储对象的交叉引用信息&#xff0c;就像交叉引用表的功能一样。 采用交…

C遗漏知识(个人向)

之前C语言遗漏的一些。 数据在内存中的存储 原码、反码、补码 整数的2进制表⽰⽅法有三种&#xff0c;即 原码、反码和补码 正整数的原、反、补码都相同。 负整数的三种表⽰⽅法各不相同。 原码&#xff1a;直接将数值按照正负数的形式翻译成⼆进制得到的就是原码。 反码&…

Javascript入门学(基础)

软件篇 JS基础语法第一天 1.javascript介绍 1.1 js是什么 是什么 是一种运行在客户端&#xff08;浏览器&#xff09;的编程语言&#xff0c;实现人机交互效果&#xff0c;而html和css是标记性语言&#xff0c;并非编程语言有什么用 js的组成 htmlcssjs实现按钮点击功能 …

Vision Transfomer系列第一节---从0到1的源码实现

本专栏主要是深度学习/自动驾驶相关的源码实现,获取全套代码请参考 这里写目录标题 准备逐步源码实现数据集读取VIt模型搭建hand类别和位置编码类别编码位置编码 blocksheadVIT整体 Runner(参考mmlab)可视化 总结 准备 本博客完成Vision Transfomer(VIT)模型的搭建和flowers数…

股票K线简介

股票K线&#xff08;K-Line&#xff09;是用于表示股票价格走势的图形&#xff0c;主要由四个关键价格点组成&#xff1a;开盘价、收盘价、最高价和最低价。K线图广泛应用于股票市场技术分析中&#xff0c;它提供了丰富的信息&#xff0c;帮助分析师和投资者理解市场的行情走势…

一周学会Django5 Python Web开发-Django5介绍及安装

锋哥原创的Python Web开发 Django5视频教程&#xff1a; 2024版 Django5 Python web开发 视频教程(无废话版) 玩命更新中~_哔哩哔哩_bilibili2024版 Django5 Python web开发 视频教程(无废话版) 玩命更新中~共计10条视频&#xff0c;包括&#xff1a;2024版 Django5 Python we…

算法练习-四数之和(思路+流程图+代码)

难度参考 难度&#xff1a;中等 分类&#xff1a;数组 难度与分类由我所参与的培训课程提供&#xff0c;但需要注意的是&#xff0c;难度与分类仅供参考。且所在课程未提供测试平台&#xff0c;故实现代码主要为自行测试的那种&#xff0c;以下内容均为个人笔记&#xff0c;旨在…