第九十三天学习记录:C++核心:类和对象Ⅱ(五星重要)

对象的初始化和清理

构造函数和析构函数

对象的初始化和清理也是两个非常重要的安全问题
一个对象或者变量没有初始化状态,对其使用后果是未知
同样的使用完一个对象或变量,没有及时清理,也会造成一定的安全问题

c++利用了构造函数和析构函数解决上述问题,这两个函数将会被编译器自动调用,完成对象初始化和清理工作。
对象的初始化和清理工作是编译器强制要我们做的事情,因此如果我们不提供构造和析构,编译器会提供编译器提供的构造函数和析构函数,只是他们都是空实现。

1、构造函数:主要作用在于创建对象时为对象的成员属性赋值,构造函数由编译器自动调用,无须手动调用。
2、析构函数:主要作用在于对象销毁前系统自动调用,执行一些清理工作。

构造函数语法:类名(){}

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

构造函数语法:~类名(){}

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

person.h

#pragma once
#include<iostream>
using namespace std;
class Person
{
public:
	//  1、构造函数,没有返回值也不写void
	//	2、函数名称与类名相同
	//	3、构造函数可以有参数,因此可以发生重载
	//	4、程序在调用对象时候会自动调用构造,无须手动调用,而且只会调用一次
	Person();
	//  1、析构函数,没有返回值也不写void
	//	2、函数名称与类名相同,在名称前加上符号~
	//	3、析构函数不可以有参数,因此不可以发生重载
	//	4、程序在对象销毁前会自动调用析构,无须手动调用,而且只会调用一次
	~Person();
	int m_num=0;
	void setnum(int num);
};

main.cpp

#define _CRT_SECURE_NO_WARNINGS 1

#include "person.h"

void test01()
{
	Person p1;//在栈上的数据,test01执行完毕后,释放这个对象
	p1.setnum(1);
}

int main()
{
	Person p2;
	p2.setnum(2);
	test01();
	system("pause");
	return 0;
}

person.cpp

#define _CRT_SECURE_NO_WARNINGS 1
#include "person.h"

Person::Person()
{
	cout << "Person构造函数的调用" << m_num << endl;
}

Person::~Person()
{
	cout << "Person析构函数的调用" << m_num << endl;
}

void Person::setnum(int num)
{
	m_num = num;
}

在这里插入图片描述

构造函数的分类及调用

两种分类方式:
按参数分为:有参构造和无参构造
按类型分为:普通构造和拷贝构造
三种调用方式:
括号法
显示法
隐式转换法

#define _CRT_SECURE_NO_WARNINGS 1

#include<iostream>
using namespace std;

class  Person
{
public:
	Person()
	{
		cout << "Person的无参数构造函数调用" << endl;//无参构造函数(默认构造函数)

	}
	Person(int a)
	{
		age = a;
		cout << "Person的带参数构造函数调用" << endl;//有参构造函数
	}
	//拷贝构造函数
	Person(const Person &p)
	{
		age = p.age;
		age = p.age;
		cout << "Person的拷贝数构造函数调用" << endl;
	}
	~Person()
	{
		cout << "Person的析构函数调用" << endl;
	}

	int age;
};

void test01()
{
	//1、括号法
	//Person p1;//默认构造函数调用
	//Person p2(10);//有参构造函数
	//Person p3(p2);
	//注意事项
	//调用默认构造函数时,不要加()
	//因为编译器会认为是一个函数的声明,不会认为在创建对象
	//cout << "p2的年龄为:" << p2.age << endl;
	//cout << "p3的年龄为:" << p3.age << endl;
	//2、显示法
	//Person p1;//默认构造函数调用
	//Person p2 = Person(10);//有参构造函数
	//Person p3 = Person(p2);//拷贝构造函数
	//注意事项
	//不要利用拷贝构造函数,初始化匿名对象

	//Person(10);//匿名对象 特点:当前行执行结束后,系统会立即回收掉匿名对象
	//3、隐式转换法
	Person p4 = 10;//相当于写了 Person p4 = Person(10);//有参构造函数
	Person p5 = p4;
}

int main()
{
	test01();
	return 0;
}

拷贝构造函数的调用时机

C++中拷贝构造函数调用时机通常有三种情况
1、使用一个已经创建完毕的对象来初始化一个新对象
2、值传递的方式给函数传值
3、以值方式返回局部对象

#define _CRT_SECURE_NO_WARNINGS 1

#include<iostream>
using namespace std;

class  Person
{
public:
	Person()
	{
		cout << "Person的无参数构造函数调用" << endl;//无参构造函数(默认构造函数)

	}
	Person(int age)
	{
		m_Age = age;
		cout << "Person的带参数构造函数调用" << endl;//有参构造函数
	}
	//拷贝构造函数
	Person(const Person &p)
	{
		m_Age = p.m_Age;
		cout << "Person的拷贝数构造函数调用" << endl;
	}
	~Person()
	{
		cout << "Person的析构函数调用" << endl;
	}

	int m_Age;
};

//1、使用一个已经创建完毕的对象来初始化一个新对象
void test01()
{
	Person p1(20);
	Person p2(p1);
}

//2、值传递的方式给函数传值
void doWork(Person p)
{

}

void test02()
{
	Person p;
	doWork(p);
}

//3、以值方式返回局部对象
Person doWork2()
{
	Person p1;
	return p1;
}

void test03()
{
	Person p = doWork2();
}

int main()
{
	//test01();
	//test02();
	test03();
	system("pause");
	return 0;
}

自行探索:☆

1、当类中有任意一个构造函数时,编译器就不会在添加默认构造函数了。
2、当类中程序员自己写了一个拷贝构造函数后,若这个拷贝函数中并没有将所有成员变量的赋值给另一个需要拷贝的类,那么在调用值传递的时候,这些常量的值将不会被拷贝过去。

构造函数的调用规则

默认情况下,c++编译器至少给一个类添加3个函数
1、默认构造函数(无参,函数体为空)
2、默认析构函数(无参,函数体为空)
3、默认拷贝构造函数,对属性进行值拷贝

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

#include<iostream>
using namespace std;

class  Person
{
public:
	Person()
	{
		cout << "Person的无参数构造函数调用" << endl;//无参构造函数(默认构造函数)
	}
	Person(int age)
	{
		m_Age = age;
		cout << "Person的带参数构造函数调用" << endl;//有参构造函数
	}
	//Person(const Person &p)//*(Person* const p)
	//{
	//	m_Age = p.m_Age;
	//	cout << "Person的拷贝数构造函数调用" << endl;//拷贝构造函数
	//}
	~Person()
	{
		cout << "Person的析构函数调用" << endl;
	}
	int m_Age;
};

void test01()
{
	Person p;
	p.m_Age = 18;
	Person p2(p);
	cout << "p2的年龄为:" << p2.m_Age << endl;
}

int main()
{
	test01();
	system("pause");
	return 0;
}
#include<iostream>
using namespace std;

class  Person
{
public:
	//Person()
	//{
	//	cout << "Person的无参数构造函数调用" << endl;//无参构造函数(默认构造函数)
	//}
	Person(int age)
	{
		m_Age = age;
		cout << "Person的带参数构造函数调用" << endl;//有参构造函数
	}
	//Person(const Person &p)//*(Person* const p)
	//{
	//	m_Age = p.m_Age;
	//	cout << "Person的拷贝数构造函数调用" << endl;//拷贝构造函数
	//}
	~Person()
	{
		cout << "Person的析构函数调用" << endl;
	}
	int m_Age;
};

//void test01()
//{
//	Person p;//报错
//	p.m_Age = 18;
//	Person p2(p);
//	cout << "p2的年龄为:" << p2.m_Age << endl;
//}

void test02()
{
	Person p(28);
	Person p2(p);
	cout << "p2的年龄为:" << p2.m_Age << endl;
}

int main()
{
	//test01();
	test02();
	system("pause");
	return 0;
}

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

React Native 集成到iOS原有的项目上

1.官方说明地址&#xff1a; 集成到现有原生应用 2. 主要步骤说明 把 React Native 组件集成到 iOS 应用中有如下几个主要步骤&#xff1a; 配置好 React Native 依赖和项目结构。了解你要集成的 React Native 组件。使用 CocoaPods&#xff0c;把这些组件以依赖的形式加入到项…

机器学习复习6

机器学习复习 1 - 在机器学习的背景下&#xff0c;什么是诊断(diagnostic)&#xff1f; A. 这指的是衡量一个学习算法在测试集(算法没有被训练过的数据)上表现如何的过程 B. 迅速尝试尽可能多的不同方法来改进算法&#xff0c;从而看看什么方法有效 C. 机器学习在医疗领域的应用…

Python 基本数据类型(五)

文章目录 每日一句正能量List&#xff08;列表&#xff09;结语 每日一句正能量 营造良好的工作和学习氛围&#xff0c;时刻牢记宗旨&#xff0c;坚定信念&#xff0c;胸怀全局&#xff0c;埋头苦干&#xff0c;对同事尊重信任谅解&#xff0c;发扬团体协作精神&#xff0c;积极…

让集合数据操控指尖舞动:迭代器和生成器的精妙之处

文章目录 &#x1f499;迭代器&#xff08;Iterator&#xff09;迭代器的特点&#xff1a;迭代器的优点&#xff1a;代码案例&#xff1a; &#x1f49a;生成器&#xff08;Generator&#xff09;生成器的特点&#xff1a;生成器的优点&#xff1a;代码案例&#xff1a; &#…

在WSL2中安装IntelliJ IDEA开发工具

一、wsl支持图形 windows安装xming https://sourceforge.net/projects/xming/ 添加白名单 查看服务器ip ifconfig 编辑配置文件(结合自己的安装目录) ‪D:\ProgramFiles\Xming\X0.hosts 启动Xlaunh wsl 配置并验证 #b编辑配置文件 vi ~/.bashrc #末尾增加配置 export DI…

二、1什么是面向对象编程?

你好&#xff0c;我是程序员雪球&#xff0c;接下来与你一起学习什么是面向对象编程。 面向对象编程是一种编程风格。它以类或对象作为组织代码的基本单元&#xff0c;并将封装&#xff0c;抽象&#xff0c;继承&#xff0c;多态四个特性&#xff0c;作为代码设计的实现基石。 …

你如何理解 JS 的继承?

在JavaScript中&#xff0c;继承是一种机制&#xff0c;允许一个对象&#xff08;子类&#xff09;从另一个对象&#xff08;父类&#xff09;继承属性和方法。这使得子类可以共享父类的功能&#xff0c;并有能∧自身定义新的功能。 JavaScript中的继承通过原型链实现。 具体来…

RabbitMQ管理界面介绍

1.管理界面概览 connections&#xff1a; 无论生产者还是消费者&#xff0c;都需要与RabbitMQ建立连接后才可以完成消息的生产和消费&#xff0c;在这里可以查看连接情况 channels&#xff1a; 通道&#xff0c;建立连接后&#xff0c;会形成通道&#xff0c;消息的投递获取依…

ChatGPT中 top_p 和 temperature 的作用机制

1. temperature 的作用机制 GPT 中的 temperature 参数调整模型输出的随机性。随机性大可以理解为多次询问的回答多样性、回答更有创意、回答更有可能没有事实依据。随机性小可以理解为多次询问更有可能遇到重复的回答、回答更接近事实&#xff08;更接近训练数据&#xff09;…

自动化测试框架[Cypress概述]

目录 前言&#xff1a; Cypress简介 Cypress原理 Cypress架构图 Cypress特性 各类自动化测试框架介绍 Selenium/WebDriver Karma Karma的工作流程 Nightwatch Protractor TestCafe Puppeteer 前言&#xff1a; Cypress是一个基于JavaScript的端到端自动化测试框架…

【SpringMVC 学习笔记】

SpringMVC 笔记记录 1. SpringMVC 简介2. 入门案例3. 基本配置3.1 xml形式配置3.2 注解形式配置 4. 请求4.1 请求参数4.1.1 普通类型传参4.1.2 实体类类型传参4.1.3 数组和集合类型传参 4.2 类型转换器4.3 请求映射 5. 响应 1. SpringMVC 简介 三层架构 2. 入门案例 3. 基本…

基于matlab使用深度学习估计身体姿势(附源码)

一、前言 此示例演示如何使用 OpenPose 算法和预训练网络估计一个或多个人的身体姿势。 身体姿势估计的目标是识别图像中人的位置及其身体部位的方向。当场景中存在多个人时&#xff0c;由于遮挡、身体接触和相似身体部位的接近&#xff0c;姿势估计可能会更加困难。 有两种…

Spring概念:容器、Ioc、DI

目录 什么是容器&#xff1f; 什么是 IoC&#xff1f; 传统程序的开发 理解 Spring IoC DI 总结 我们通常所说的 Spring 指的是 Spring Framework&#xff08;Spring 框架&#xff09;&#xff0c;它是⼀个开源框架&#xff0c;有着活跃⽽庞⼤的社区&#xff0c;这就是它…

Swin Transformer训练报错问题

1. 训练遇到报错问题 &#xff08;1&#xff09;mportError: cannot import name _pil_interp from timm.data.transforms 原因&#xff1a; timm.data.transforms里面没有_pil_interp&#xff0c;只有str_to_pil_interp、_str_to_pil_interpolation、_pil_interpolation_to_s…

【Docker】docker安装配置Jenkins

docker 安装 Jenkins #拉镜像 docker pull jenkins/jenkins#创建卷(volume) docker volume create jenkins_home#制作容器并启动 docker run -d \ -p 8080:8080 \ -p 50000:50000 \ -v jenkins_home:/var/jenkins_home \ -v /usr/lib/jvm/java-8-openjdk-amd64:/usr/local/java…

如何将window文件夹挂载到VMware系统mnt目录

背景&#xff1a;项目开发过程中&#xff0c;通常是在Windows上编码&#xff0c;有些框架和软件只能够在Linux上面执行&#xff0c;如果在 VMware中的Linux上面开发不太方便&#xff0c;因此需要在Windows上面开发好再同步到Linux上面运行。 软件&#xff1a; Samba客户端 V…

配置Jenkins的slave agent并使用它完成构建任务

上一章&#xff0c;使用单机配置并运行了一个简单的maven项目&#xff0c;并发布到了一个服务器上启动。这一章将要配置一个slave agent&#xff0c;并将上一章的job放到agent上执行。我们agent使用的是ssh的方式 前置步骤 准备两台虚拟机&#xff1a; 192.168.233.32&#…

svn commit 用法

转载   原文&#xff1a;https://blog.csdn.net/qq_39790633/article/details/103700391 使用svn进行代码的提交有两种方法&#xff1a;一种是通过TortoiseSVN客户端界面进行提交&#xff0c;另一种是通过svn commit指令提交。 方法一&#xff1a;通过TortoiseSVN客户端界面提…

STM32速成笔记—IWDG

文章目录 一、IWDG简介二、STM32的IWDG2.1 STM32的IWDG简介2.2 喂狗2.3 IWDG框图 三、IWDG配置步骤四、IWDG配置程序4.1 IWDG初始化程序4.2 喂狗 五、应用实例 一、IWDG简介 独立看门狗&#xff08;Independent Watchdog, IWDG&#xff09;&#xff0c;什么是看门狗&#xff1…

NVIDIA-Linux-x86_64-535.54.03.run cuda_12.2.0_535.54.03_linux.run下载地址

Official Drivers | NVIDIA Linux x64 (AMD64/EM64T) Display Driver | 535.54.03 | Linux 64-bit | NVIDIA 下载连接 Download NVIDIA, GeForce, Quadro, and Tesla DriversDownload drivers for NVIDIA graphics cards, video cards, GPU accelerators, and for other GeFor…