第六站:C++面向对象关键字解释说明

  this指针:

是一个特殊的指针,放回这个对象本身,this指针是属于实例对象,不能访问静态方法(不属于某一个实例对象,属于共有的,大众的,由类直接调用)

第一种用法:

void Human::setName(string name1) {
	this->name = name1;
}
void Human::setAge(int age1) {
	this->age = age1;
}
void Human::setSalary(int salary1) {
	this->salary = salary1;
}

 第二三种用法:引用和指针

//this指针的用法两种
const Human* comBetterMan(const Human*);
const Human& comBetterMan1(const Human&);

/*
//当返回对象是指针时
const Human *Human::comBetterMan(const Human *other) {
	if (this->getSalary() >other->getSalary()) {
		return this;
	}
	else {
		return other;
	}
}
//当返回对象是引用时
const Human& Human::comBetterMan1(const Human&other) {
	if (this->getSalary() > other.getSalary()) {
		return *this;
	}
	else {
		return other;
	}
}
*/
int main(void) {
	//自定义一个带参的构造函数
	Human h1("张三", 24, 25000);
	Human h2("李四", 40, 50000);

	cout << "用指针返回的是:" << h1.comBetterMan(&h2) << endl;
	cout << "用引用返回的是:" << &h1.comBetterMan1(h2) << endl;
	return 0;
}

类文件的分离

 分为定义(.h的文件中)和实现(.cpp文件中)两个部分和一个主函数(main)文件.cpp

.h中

#pragma once
#include <iostream>
#include <string>
using namespace std;

class Human {
public://公有的,对外的
	void eat();//方法,成员函数
	void sleep();
	void play();
	void work();
	void descript();
	//自定义的默认构造函数
	Human();
	~Human();
	//手动定义带参的(重载)构造函数
	Human(string name, int age, int salary);
	//手动定义拷贝构造函
	Human(const Human& other);
	//手动定义赋值构造函数,
	// 运算符重载"operator ="
	Human& operator= (const Human& other1);
	void setName(string name);//设置对内的数据
	void setAge(int age);
	void setSalary(int salary);
	void setAddr(char* addr);
	//this指针的用法两种
	const Human* comBetterMan(const Human*);
	const Human& comBetterMan1(const Human&);

	string getName()const;//获取对内的数据
	int getAge()const;
	int getSalary()const;
	char* getAddr() const;
private://对内的,私有的
	string name;
	int age;//设置类内初始值
	int salary;
	char* addr;
};

实现 .cpp中代码


#include "Human.h"
#define size 16
string Human::getName() const {
	return name;
}
int Human::getAge() const {
	return age;
}
int Human::getSalary() const {
	return salary;
}
char* Human::getAddr() const
{
	return addr;
}
//手动定义的默认构造函数
Human::Human() {
	cout << "手动定义的默认构造函数" << this << endl;
	name = "张三";
	age = 23;
	salary = 25000;
	addr = new char[32];
	strcpy_s(addr, size, "Amecrican");
}
//自定义重载构造函数
Human::Human(string name, int age, int salary) {
	cout << "调用自定义的带参构造函数" << this << endl;
	this->age = age;
	this->name = name;
	this->salary = salary;
	addr = new char[32];
	strcpy_s(addr, size, "China");
}
//相当于 Human h2 = h1;    //const Human &other = h1;
Human::Human(const Human& other) {
	cout << "调用拷贝构造函数" << endl;
	this->age = other.age;
	this->name = other.name;
	this->salary = other.salary;
	//为addr分配一块独立的内存空间,不进行浅拷贝
	addr = new char[32];
	strcpy_s(addr, size, other.addr);//设置值为深拷贝
}

//如果时引用的话,是不会创建新的对象
//void type2(const Human& h2) {//相当于别名
//	cout << "拷贝构造函数作为函数的参数,是引用类型" << endl;
//	//h2.setAddr("芜湖");//可以修改值,定义为const就不能修改了
//	cout << "name: " << h2.getName()
//		<< "age: " << h2.getAge()
//		<< "salary: " << h2.getSalary()
//		<< "addr: " << h2.getAddr() << endl;
//}
//函数的返回类型是类,而非引用类型,,参数是对象,而非引用类型
//这个函数会返回一个临时的对象,这个对象会调用拷贝构造函数,加上参数的两个,这里一共会调用三次拷贝构造函数
//这里可以将参数设置为引用类型,避免浪费资源(其他成员函数最好设置为const的类型,避免值无意间自己给修改了)
//返回的类型也可以设置为引用类型,避免创建临时对象
const Human& getBetterMan(const Human& h1, const Human& h2) {
	if (h1.getSalary() > h2.getSalary()) {
		return h1;
	}
	else {
		return h2;
	}
}
//赋值构造函数,&Human等同于h1
//const型的   h2
Human& Human::operator= (const Human& other1) {
	cout << "调用赋值构造函数" << this << endl;
	//防止对象给自己赋值
	if (this == &other1) {
		//如果对象给自己赋值,就返回对象本身的指针
		return *this;
	}
	//当执行h2 = h1时 
	//就相当于调用h2.operator =(h1);
	this->name = other1.name;
	this->age = other1.age;
	this->salary = other1.salary;
	//如果有必要可以先释放原本对象的资源
	//看情况,因为对象生成的时候本身就有一块内存
	delete addr;
	addr = new char[64];
	strcpy_s(this->addr, size, other1.addr);//深拷贝
	//返回对象本身的引用,以方便做链式处理,h3=h2=h1;
	return *this;//return this 这种只是返回对象的指针,这个对象如果继续给下一个对象赋值的,就是空的
}
void Human::setName(string name1) {
	this->name = name1;
}
void Human::setAge(int age1) {
	this->age = age1;
}
void Human::setSalary(int salary1) {
	this->salary = salary1;
}
void Human::setAddr(char* addr)
{
	if (!addr) {
		return;
	}
	strcpy_s(this->addr, size, addr);
}
//当返回对象是指针时
const Human* Human::comBetterMan(const Human* other) {
	if (this->getSalary() > other->getSalary()) {
		return this;
	}
	else {
		return other;
	}

}

//当返回对象是引用时
const Human& Human::comBetterMan1(const Human& other) {
	if (this->getSalary() > other.getSalary()) {
		return *this;
	}
	else {
		return other;
	}
}
Human::~Human() {
	//这里加个this,谁来调用就返回这个对象的指针,在其他对象中也加入就可以判断调用的先后顺序
	cout << "调用析构函数" << this << endl;
	delete addr;
}

 主函数.cpp

#include <iostream>
#include <string>
#include "Human.h"
using namespace std;
#define size 16
//成员描述信息
void Human::descript() {
	cout << "name: " << getName()
		<< "age: " << getAge()
		<< "salary: " << getSalary()
		<< "addr: " << addr << endl;//可以直接访问,只是不安全
}

//拷贝构造函数作为函数的参数,而非引用类型
void type1(Human h2) {
	cout << "拷贝构造函数作为函数的参数,而非引用类型" << endl;
	cout << "name: " << h2.getName()
		<< "age: " << h2.getAge()
		<< "salary: " << h2.getSalary()
		<< "addr: " << h2.getAddr() << endl;
}

int main(void) {
	//自定义一个带参的构造函数
	Human h1("张三", 24, 25000);
	Human h2("李四", 40, 50000);

	cout << "用指针返回的是:" << h1.comBetterMan(&h2) << endl;
	cout << "用引用返回的是:" << &h1.comBetterMan1(h2) << endl;
	return 0;
}

static关键字

弥补使用全局变量不方便,破坏程序的封装性,就是用静态

对于非 const 的类静态成员

只能在类的实现文件中初始化。所有成员函数都可读可写

头文件.h
class Human{

public:
......
private:
....
static int count;
....
};

实现文件.cpp

//初始化静态成员
//对于非 const 的类静态成员,只能在类的实现文件中初始化。
int Human::count = 0;

 const 类静态成员

可以在类内设置初始值,也可以在类的实现文件中设置初始值。(但是 不要同时在这两个地方初始化,只能初始化 1 次)

类的静态方法:别人可以访问他,但是他不能访问别人

优点:可以节省创建对象来调用成员函数,因为静态的成员函数可以用类来调用

1. 可以直接通过类来访问【更常用】,也可以通过对象(实例)来访问

        / 直接通过类名来访问静态方法!
        // 用法:类名::静态方法

2. 在类的静态方法中,不能访问普通数据成员和普通成员函数(对象的数据成员和成员函

数)

静态方法中,只能访问静态数据成员

静态方法的实现,不能加 static

#pragma once
......
class Human {
public:
......
static int getCount();
......
};

实现文件中
//静态方法的实现,不能加 static
int Human::getCount() {
// 静态方法中,不能访问实例成员(普通的数据成员)
// cout << age;
// 静态方法中,不能访问 this 指针,因为 this 指针是属于实例对象的
// cout << this;
//静态方法中,只能访问静态数据成员
return count;
}

//静态方法:别人可以访问他,但是他不能访问别人
void test() {
cout << "总人数: ";
// ??? 没有可用的对象来访问 getCount()
// 直接通过类名来访问静态方法!
// 用法:类名::静态方法
cout << Human::getCount();
}



int main(void) {
	//自定义一个带参的构造函数
	Human h1("张三", 24, 25000);
	Human h2("李四", 40, 50000);

	//可以通过类名访问静态成员方法,也可以通过实例对象
	cout << Human::getCount() << endl;
	cout << h1.getCount() << endl;
	return 0;
}

所有的成员函数,都可以访问静态数据成员。

类可以直接访问public静态数据成员(Human::count 非法),但是不能访问私有的静态数据成员

对象可以直接访问静态成员函数
类可以直接访问静态成员函数(但是类不能访问非静态成员函数)
在类的静态成员函数(类的静态方法)内部, 不能直接访问 this 指针和对象的数据成员

 const关键字

const成员数据

只能读不能写

初始化:

设置类内初始值,(C11以上版本支持)

const char BloodType = 'B';

使用构造函数的初始化列表:一个冒号+const成员('初始值')

Human::Human(string name, int age, int salary):BloodType('A') 

 如何做到灵活初始化,用带参的构造函数的参数列表里定义一个形参,去表示初始化列表里的值

/*
Human(string name, int age, int salary,string BloodType);
const string BloodType = "B";
*/
实现文件
//在定义的对象的时候就可以灵活的设置初始值了
Human::Human(string name, int age, int salary,string BloodType):BloodType(BloodType) {
}


int main(void) {
	//自定义一个带参的构造函数
	Human h1("张三", 24, 25000,"AB");
	Human h2("李四", 40, 50000,"A");

	h1.descript();
	h2.descript();
	return 0;
}

const成员函数

如果一个成员函数内部,不会修改任何数据成员,就把它定义为 const 成员函数。
const方法只能调用const方法,非const可以调用const方法
void descript() const;
//成员描述信息
void Human::descript() const{//这个函数里面就不能修改数据
	cout << "name: " << getName()
		<< "age: " << getAge()
		<< "salary: " << getSalary()
		<< "addr: " << addr 
		<< "bloodType:"<<BloodType << endl;//可以直接访问,只是不安全
}

int main(void) {
	//自定义一个带参的构造函数
	const Human h1("张三", 24, 25000,"AB");//const对象可以访问const的成员函数
	Human h2("李四", 40, 50000,"A");//也可以访问const的成员函数

	h1.descript();
	h2.descript();

	return 0;
}

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

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

相关文章

拯救者y9000p安装linux、windows双系统。

首先需要准备启动盘 我用的是Win32DiskImager来做的。资源使用的是ubuntu-20.04.6-desktop-amd64.iso。别用低版本&#xff0c;失败很多次之后的教训。 磁盘管理-磁盘分区-右键-压缩卷 这边分区出来之后&#xff0c;不要分配。安装时候会自动分配的。 重启之后F2进去BIOS设置…

张载为往圣继绝学,唯一的错是不够强大

“自古雄才多磨难&#xff0c;从来纨绔少伟男。” 张载&#xff0c;人称“横渠先生”。他在横渠镇&#xff0c;授徒讲学&#xff0c;恢复古礼&#xff0c;试验井田&#xff0c;写书《正蒙》。张载讲学关中&#xff0c;弟子多为关中人&#xff0c;其学派被称作关学。 张载自学…

git-生成证书、公钥、私钥、error setting certificate verify locations解决方法

解决方法 方法1-配置证书、公钥、私钥打开Git Bash设置名称和邮箱执行&#xff0c;~/.ssh执行&#xff0c;ssh-keygen -t rsa -C "这是你的邮箱"&#xff0c;如图&#xff1a;进入文件夹可以看到用记事本之类的软件打开id_rsa.pub文件&#xff0c;并且复制全部内容。…

Python自动化测试框架【生成测试报告】

如何才能让用例自动运行完之后&#xff0c;生成一张直观可看易懂的测试报告呢&#xff1f; 小编使用的是unittest的一个扩展HTMLTestRunner 环境准备 使用之前&#xff0c;我们需要下载HTMLTestRunner.py文件 点击HTMLTestRunner后进入的是一个写满代码的网页&#xff0c;小…

并发编程之并发容器

目录 并发容器 CopyOnWriteArrayList 应用场景 常用方法 读多写少场景使用CopyOnWriteArrayList举例 CopyOnWriteArrayList原理 CopyOnWriteArrayList 的缺陷 扩展迭代器fail-fast与fail-safe机制 ConcurrentHashMap 应用场景 常用方法 并发场景下线程安全举例 Con…

软件测试|教你使用Python下载图片

前言 我一直觉得Windows系统默认的桌面背景不好看&#xff0c;但是自己又没有好的资源可以进行替换&#xff0c;突然我一个朋友提醒了我&#xff0c;网络上的图片这么多&#xff0c;你甚至可以每天换很多个好看的背景&#xff0c;但是如果让我手动去设置的话&#xff0c;我觉得…

编程高手必备:Python字典操作与示例全解析

编程高手必备&#xff1a;Python字典操作与示例全解析 引言Python字典基础字典操作技巧代码示例实际应用案例实际应用案例结语 引言 在编程世界中&#xff0c;高效地处理和组织数据是每位程序员必备的技能。Python&#xff0c;作为一种广受欢迎的编程语言&#xff0c;不仅因其…

mysql原理--undo日志1

1.事务回滚的需求 我们说过 事务 需要保证 原子性 &#xff0c;也就是事务中的操作要么全部完成&#xff0c;要么什么也不做。但是偏偏有时候事务执行到一半会出现一些情况&#xff0c;比如&#xff1a; (1). 事务执行过程中可能遇到各种错误&#xff0c;比如服务器本身的错误&…

Jenkins-用户管理

用户管理 1 安装插件 2 选择安全策略为刚刚安装的插件 3 这个是安装插件以后会有的选项 4 增加一个角色 5 根据需要赋值角色的权限&#xff0c;并分配给用户

橘子学Mybatis07之Mybatis关于缓存的设计

很逆天的一件事是&#xff0c;我上一次发mybatis是在2022年10月15号&#xff0c;然后直到今天才开始总结下一篇Mybatis的东西。一年里面忙成那啥了&#xff0c;而且重心都投入在了Elasticsearch的学习上面&#xff0c;基本一年下来都在搞ES&#xff0c;并且考下了ECE认证&#…

【Docker】Dockerfile构建最小镜像

&#x1f973;&#x1f973;Welcome 的Huihuis Code World ! !&#x1f973;&#x1f973; 接下来看看由辉辉所写的关于Docker的相关操作吧 目录 &#x1f973;&#x1f973;Welcome 的Huihuis Code World ! !&#x1f973;&#x1f973; 前言 一.Dockerfile是什么 二.Dock…

lenovo联想笔记本电脑拯救者Legion Y7000 2019 PG0(81T0)原装出厂Windows10系统

链接&#xff1a;https://pan.baidu.com/s/1fn0aStc4sfAfgyOKtMiCCA?pwdas1l 提取码&#xff1a;as1l 联想拯救者原厂Win10系统自带所有驱动、出厂主题壁纸、系统属性联机支持标志、系统属性专属LOGO标志、Office办公软件、联想电脑管家等预装程序 所需要工具&#xff1a;…

模型索引:QModelIndex

一、为什么要使用模型索引&#xff1f; 从名字可以看出&#xff0c;他是模型的索引&#xff0c;只要对模型实体&#xff08;各种xxxModel的实体&#xff09;施加这个索引&#xff0c;model就会返回数据集中对应的值&#xff0c;或者通过这个索引修改对应数据集中的值。 类比数…

VMware安装CentOS7虚拟机

VMware 安装 获取 VMware 安装包 下载地址&#xff1a;链接&#xff1a;https://pan.baidu.com/s/1ELR5NZa7rO6YVplZ1IUigw?pwdplz3 提取码&#xff1a;plz3 包括&#xff1a;当然&#xff0c;也可以自己去别的地方下载&#xff0c;WMware 版本都差不多&#xff0c;现在用的比…

数据结构学习 jz14剪绳子

关键词&#xff1a;数学 动态规划 快速幂 这道题其实是分为两题。 题目一&#xff1a; 这道题我是没有思路的&#xff0c;看了k神的答案才知道有数学的方法。 方法一&#xff1a; 数学&#xff1a;其实中间的推导我没看&#xff0c;我服了。 思路&#xff1a; 复杂度计算&…

matplotlib:热图、箱形图、小提琴图、堆叠面积图、雷达图、子图

简介&#xff1a;在数字化的世界里&#xff0c;从Web、HTTP到App&#xff0c;数据无处不在。但如何将这些复杂的数据转化为直观、易懂的信息&#xff1f;本文将介绍六种数据可视化方法&#xff0c;帮助你更好地理解和呈现数据。 热图 (Heatmap)&#xff1a;热图能有效展示用户…

WorkPlus企业内部即时通信新选择,打造高效协作新格局

在企业内部&#xff0c;快速、高效的沟通与协作是推动工作进程的关键。而即时通信工具成为了企业内部沟通的重要工具。作为一款优秀的企业内部即时通信工具&#xff0c;WorkPlus通过其出色的性能和独特的功能&#xff0c;为企业打造高效协作的新格局。 为什么选择WorkPlus作为企…

详解Matlab深度学习进行波形分割

&#x1f517; 运行环境&#xff1a;Matlab &#x1f6a9; 撰写作者&#xff1a;左手の明天 &#x1f947; 精选专栏&#xff1a;《python》 &#x1f525; 推荐专栏&#xff1a;《算法研究》 &#x1f510;#### 防伪水印——左手の明天 ####&#x1f510; &#x1f497; 大家…

强化学习应用(三):基于Q-learning算法的无人车配送路径规划(提供Python代码)

一、Q-learning算法介绍 Q-learning是一种强化学习算法&#xff0c;用于解决基于环境的决策问题。它通过学习一个Q-table来指导智能体在不同状态下采取最优动作。下面是Q-learning算法的基本步骤&#xff1a; 1. 定义环境&#xff1a;确定问题的状态和动作空间&#xff0c;并…

逆变器之推挽谐振

首先把前级推挽电路分成几个模块&#xff1a;方波发生器、谐振LC、整流滤波以及负载。框图如下图所示&#xff1a; 分析前提&#xff1a;稳态 在推挽电路正常工作中&#xff0c;输入电压恒定、输出电流电压也恒定&#xff08;电源处于稳定的工作状态中&#xff09; 方波发生器…