重生学c++系列第三课类和对象(上)

news2025/1/14 2:14:11

    好的我们重生c++系列的前两期已经介绍完了c++祖师爷针对C语言补充的几个新功能,现在我们进入c++的真正课题学习——类与对象:

    C语言是面向过程的,关注的是过程,分析出求解问题的步骤,通过函数调用逐步解决问题。

    比如说我们洗菜做饭,C语言关注的是 拿盆子  倒水 洗菜  放油 炒菜 等等等这些细致的步骤

   

    C++是基于面向对象的,关注的是对象,将一件事情拆分成不同的对象,靠对象之间的交互完
成。

     一共有四个对象 :人  菜   锅  盘子  我们不需要关心菜是如何炒出来的

 类的引入

     

C语言结构体中只能定义变量,在C++中,结构体内不仅可以定义变量,也可以定义函数。比如:
之前在数据结构初阶中,用C语言方式实现的栈,结构体中只能定义变量;现在以C++方式实现,
会发现struct中也可以定义函数。
我们定义一个结构体
//c++兼容C语言,C语言的用法可以接着用,也有新用法
struct Stack
{
    int* a;
    int top;
    int capacity;
};
在c++中,我们更喜欢称其为类,c++兼容C语言,C语言的用法可以接着用,也有新用法
比如说:
#define _CRT_SECURE_NO_WARNINGS 1
#include<iostream>
using namespace std;
struct Stack
{
	int* a;
	int top;
	int capacity;
};
int main()
{
	struct Stack st1;
	Stack st2;

	return 0;
}

主函数中两种建立类的方式不同,第一种是C语言的玩法,第二种是c++的玩法,明显要更简便一点

同时类中不仅仅可以定义成员变量,并且可以定义成员函数,比如说初始化之类的函数,就没必要在外边写了

上代码:

#define _CRT_SECURE_NO_WARNINGS 1
#include<iostream>
using namespace std;
struct Stack
{
	//成员函数
	void Init()
	{
		a = nullptr;
		top = 0;
		capacity = 0;
	}
	//成员变量
	int* a;
	int top;
	int capacity;
};
int main()
{
	
	Stack st2;
	st2.Init();
	return 0;
}

这样调用Init初始化函数,是不是比C语言阶段在外部传参进去容易多了!

类的定义

上面结构体的定义, C++ 中更喜欢用 class 来代替,这也是真正的类与结构体不一样的地方
class className
{
// 类体:由成员函数和成员变量组成
};   // 一定要注意后面的分号
class为定义类的关键字,ClassName为类的名字,{}中为类的主体,注意类定义结束时后面分
号不能省略。
类体中内容称为类的成员:类中的变量称为类的属性或成员变量; 类中的函数称为类的方法或者
成员函数。
那么class里的使用和struct有什么区别,第一大区别就是访问限定符,class中有着访问限定符
我们用代码来向大家讲解:

 大家注意看,在将struc换成class之后,就无法访问我们在类中定义的函数了

这是因为我们没有加上访问限定符:publi private 

c++的类是很任性的,我可以选择给你看给你用,也可以选择不给你看,不像struct随便访问,想给你看的加上publi(共有),不想给你看的就加上private(私有)

#define _CRT_SECURE_NO_WARNINGS 1
#include<iostream>
using namespace std;
class Stack
{
public:
	//成员函数
	void Init()
	{
		a = nullptr;
		top = 0;
		capacity = 0;
	}
	void Push(int x)
	{
		if (top == capacity)
		{
			size_t newcapacity = capacity == 0 ? 4 : capacity * 2;
			a = (int*)realloc(a,sizeof(int) * newcapacity);
			capacity = newcapacity;
		}
		a[top++] = x;
	}
private:
	//成员变量
	int* a;
	int top;
	int capacity;
};
int main()
{
	
	Stack st2;
	st2.Init();
	st2.Push(1);
	st2.Push(2);
	st2.Push(3);
	st2.Push(4);

	return 0;
}

这样,我们在class中写的函数就可以用了,因为我们在他们的前面加上了public访问限定符

很多同学学到这里会有这样一个疑问,private限制类外访问,那限制类里的访问吗,答案是当然可以,就像是你家有一个保险柜,肯定用来防外人,自己家里人肯定知道密码的对吧。

成员变量的风格

我们定义一个新的日期的类

#define _CRT_SECURE_NO_WARNINGS 1
#include<iostream>
using namespace std;
class Date
{
public:
	void Init(int year, int month, int day)
	{
		year = year;
		month = month;
		day = day;
	}
private:
	int year;
	int month;
	int day;
};
int main()
{
	
	Date d;
	d.Init(2022,7,13);

	return 0;
}

大家看这段代码,思考一下有没有初始化成功

答案:

编译是没有任何问题的,但是:

我们调试就会发现,这三个变量都是随机值,没有被初始化

这就是我们在C语言时期讲过的局部优先原则,忘了或者没听过的小伙伴去C语言的博客里就可以找到,所以我们只需要改一下成员变量

#define _CRT_SECURE_NO_WARNINGS 1
#include<iostream>
using namespace std;
class Date
{
public:
	void Init(int year, int month, int day)
	{
		_year = year;
		_month = month;
		_day = day;
	}
private:
	int _year;
	int _month;
	int _day;
};
int main()
{
	
	Date d;
	d.Init(2022,7,13);

	return 0;
}

 这样就可以啦,这就是成员变量的风格

封装

      面向对象的三大特性:封装、继承、多态
    在类和对象阶段,主要是研究类的封装特性,那什么是封装呢?
    封装:将数据和操作数据的方法进行有机结合,隐藏对象的属性和实现细节,仅对外公开接口来
和对象进行交互。
封装本质上是一种管理,让用户更方便使用类。比如:对于电脑这样一个复杂的设备,提供给用
户的就只有开关机键、通过键盘输入,显示器,USB插孔等,让用户和计算机进行交互,完成日
常事务。但实际上电脑真正工作的却是CPU、显卡、内存等一些硬件元件。
对于计算机使用者而言,不用关心内部核心部件,比如主板上线路是如何布局的,CPU内部是如
何设计的等,用户只需要知道,怎么开机、怎么通过键盘和鼠标与计算机进行交互即可。
    因此计算机厂商在出厂时,在外部套上壳子,将内部实现细节隐藏起来,仅仅对外提供开关机、鼠标以及键盘插孔等,让用户可以与计算机进行交互即可。
    在C++语言中实现封装,可以通过类将数据以及操作数据的方法进行有机结合,通过访问权限来
隐藏对象内部实现细节,控制哪些方法可以在类外部直接被使用。  

类的作用域

类定义了一个新的作用域,类的所有成员都在类的作用域中。在类体外定义成员时,需要使用 ::
作用域操作符指明成员属于哪个类域。
class date
{
public:
 void Printdate();
private:
 char _year;
 char _month;
 int  _day;
};
// 这里需要指定Print是属于date这个类域
void date::Printdate()
{
 cout << _year << " "<< _month << " " << _day << endl;
}

类的实例化

想问大家一个问题,像下面代码这样可以访问吗?

#define _CRT_SECURE_NO_WARNINGS 1
#include<iostream>
using namespace std;
class Date
{
public:
	void Init(int year, int month, int day)
	{
		_year = year;
		_month = month;
		_day = day;
	}
private:
	int _year;
	int _month;
	int _day;
};
int main()
{
	
	Data::_year;

	return 0;
}

答案是肯定不可以,所以接着问大家一个问题,private下边的三个变量,是声明还是定义

这是一个声明,不是定义,他们两个的区别就在于声明没有开空间,所以要定义变量,开空间来使用

  1. 类是对对象进行描述的,是一个模型一样的东西,限定了类有哪些成员,定义出一个类并没
有分配实际的内存空间来存储它;比如:入学时填写的学生信息表,表格就可以看成是一个
类,来描述具体学生信息。
类就像谜语一样,对谜底来进行描述,谜底就是谜语的一个实例。
2. 一个类可以实例化出多个对象,实例化出的对象 占用实际的物理空间,存储类成员变量
3.做个比方。类实例化出对象就像现实中使用建筑设计图建造出房子,类就像是设计图,只设
计出需要什么东西,但是并没有实体的建筑存在,同样类也只是一个设计,实例化出的对象
才能实际存储数据,占用物理空间  

类的大小模型

看代码:

#define _CRT_SECURE_NO_WARNINGS 1
#include<iostream>
using namespace std;
class Date
{
public:
	void Init(int year, int month, int day)
	{
		_year = year;
		_month = month;
		_day = day;
	}
private:
	int _year;
	int _month;
	int _day;
};
int main()
{
	Date d1;
	cout << sizeof(Date) << endl;
	cout << sizeof(d1) << endl;
	return 0;
}

大家猜猜这两个的内存大小分别是多少,一个是图纸,一个是根据图纸建造的房子

小tips:大家想想C语言期间学的结构体内存对齐

答案:

有这个结果可以知道,函数是没有存在类里的

大家想想,如果我们直接定义五个类变量,难道要生成五个函数用的空间吗,大家可以把函数想象成公用的,建立的那个类变量用的函数都是一个 

类的大概存储模式

   只保存成员变量,成员函数存放在公共的代码段

每一个成员变量都是不一一样的, 但是函数放在公共代码区

结论:一个类的大小,实际就是该类中”成员变量”之和,当然要注意内存对齐
注意空类的大小,空类比较特殊,编译器给了空类一个字节来唯一标识这个类的对象

内存对齐规则

前边在C语言阶段其实我们说过内存对齐的规则,这里再给大家写一下:

1. 第一个成员在与结构体偏移量为0的地址处。
2. 其他成员变量要对齐到某个数字(对齐数)的整数倍的地址处。
注意:对齐数 = 编译器默认的一个对齐数 与 该成员大小的较小值。
VS中默认的对齐数为8
3. 结构体总大小为:最大对齐数(所有变量类型最大者与默认对齐参数取最小)的整数倍。
4. 如果嵌套了结构体的情况,嵌套的结构体对齐到自己的最大对齐数的整数倍处,结构体的整
体大小就是所有最大对齐数(含嵌套结构体的对齐数)的整数倍。

大家一定要熟记这个规则并且会运算,这个面试的时候很有可能会用到的,这里小编给大家整理的几个面试上关于内存对齐的面试题:

1. 结构体怎么对齐? 为什么要进行内存对齐?
2. 如何让结构体按照指定的对齐参数进行对齐?能否按照3、4、5即任意字节对齐?
3. 什么是大小端?如何测试某台机器是大端还是小端,有没有遇到过要考虑大小端的场景

如果有问题可以在评论区提出来

this指针

上代码:

#define _CRT_SECURE_NO_WARNINGS 1
#include<iostream>
using namespace std;
class Date
{
public:
	void Printf()
	{
		cout << _year << '/' << _month << '//' << _day << endl;
	}
    void Init(int year, int month, int day)
	{
		_year = year;
		_month = month;
		_day = day;
	}
private:
	int _year;
	int _month;
	int _day;
};
int main()
{
	Date d1;
	Date d2;
	return 0;
}

还是之前我们写的代码没有什么心意,但是大家看这一区域

我们在调用初始化函数的时候,使用的是我们在private中定义的三个变量吗?

答案:不是

我在之前说过,private里是声明,根本就没有开辟空间,并且在主函数部分我们开辟了两个变量,如果调用的是private里的变量,那么又是怎么一份变成两份的呢?所以我们根本没用这里的变量这就涉及到我们接下来要学的新知识点this指针。

void Init(int year, int month, int day)
    {
        _year = year;
        _month = month;
        _day = day;
    }

这是我们刚刚写的初始化函数,里边传了三个参数年月日完成传参赋值,其实在这个函数里,含有一个我们看不见的隐藏参数,他真正的原型应该是这样的:

void Init(Date *this ,int year, int month, int day)
    {
        this->_year = year;
        this->_month = month;
        this->_day = day;
    }

对比一下,多了个this指针,加粗的部分都是编译器自己加的

函数主体多了个参数,在在主函数调用时编译器也是自动给我们加了东西

我们自己写的:

int main()
{
    Date d1;
    d1.Init(2023, 7, 23);
    Date d2;
    d2.Init(2013, 7, 23);
    return 0;
}

实际上编译器底层编译的:

int main()
{
    Date d1;
    d1.Init(&d1,2023, 7, 23);
    Date d2;
    d2.Init(&d2,2013, 7, 23);
    return 0;
}

可以说我们可以运行代码,使用变量和函数都多亏了this指针

C++中通过引入this指针解决该问题,即:C++编译器给每个“非静态的成员函数“增加了一个隐藏
的指针参数,让该指针指向当前对象(函数运行时调用该函数的对象),在函数体中所有“成员变量”
的操作,都是通过该指针去访问。只不过所有的操作对用户是透明的,即用户不需要来传递,编
译器自动完成。

有一点值得注意,this指针在形参和实参的位置并不能由我们自己显示的写出来,但是可以在类里边显示的用

如:

class Date
{
public:
	void Init(int year, int month, int day)
	{
		this->_year = year;
		this->_month = month;
		this->_day = day;
	}
	void Printf()
	{
		cout << this->_year << '/' << this->_month << '//' << this->_day << endl;
	}
private:
	int _year;
	int _month;
	int _day;
};

像这样,但是我们不写编译器会自动给我们加,写了反而多此一举,大家了解一下就行

this指针的特性

1. this指针的类型:类类型* const,即成员函数中,不能给this指针赋值。
2. 只能在“成员函数”的内部使用
3. this指针本质上是“成员函数”的形参,当对象调用成员函数时,将对象地址作为实参传递给
this形参。所以对象中不存储this指针。
4. this指针是“成员函数”第一个隐含的指针形参,一般情况由编译器通过ecx寄存器自动传
递,不需要用户传递

 this指针的存储

我们都知道电脑的内存分为栈,堆,常量区,静态区,那么this指针存在哪里呢,大家不妨猜一猜

this指针是一个形参,一般都存在栈区

好了关于类和对象(上篇)就写到这,大家敬请期待下一期内容!!!

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

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

相关文章

QT生成Debug和Release发布版后,运行exe缺少dll问题

在QT Creator生成debug和release的exe执行文件后&#xff0c;运行时&#xff0c;报错缺少*.dll.解决办法1&#xff1a; 在系统环境变量中添加D:\Qt\Qt5.13.2\Tools\mingw730_64\bin后&#xff0c;即可运行。 当使用此方法时&#xff0c;将exe拷贝到其他电脑中运行时&#xff0c…

DAY4,C高级(shell中的函数,循环,排序思想)

1.整理思维导图&#xff1b; 2.写一个函数&#xff0c;获取用户的uid和gid并使用变量接收&#xff1b; 1 #!/bin/bash 2 function get_id()3 {4 read -p "输入用户&#xff1a;…

海外社媒营销:如何树立品牌个性与目标受众共鸣?

随着全球化的不断深入&#xff0c;海外市场对于企业的重要性越来越凸显。在这个数字化时代&#xff0c;社交媒体已经成为品牌塑造和推广的重要渠道之一。然而&#xff0c;海外市场竞争激烈&#xff0c;想要在众多品牌中脱颖而出&#xff0c;就需要在社交媒体关注者的心中树立品…

javaweb监听和渲染技术

jsp入门使用 <% page import"java.io.PrintWriter" %> <%-- Created by IntelliJ IDEA.--%> <%-- User: 韩顺平--%> <%-- jsp的模板如何定制&#xff0c;一会再说明--%> <%-- To change this template use File | Settings | File T…

64位交叉编译器版本切换

操作系统版本 lkmaoubuntu:~$ cat /proc/version Linux version 5.6.18 (lkmaoubuntu) (gcc version 7.5.0 (Ubuntu 7.5.0-3ubuntu1~18.04)) #1 SMP Tue Jul 25 23:11:17 PDT 2023 lkmaoubuntu:~$ 64位 安装默认版本和gcc5版本。 sudo apt-get install gcc-aarch64-linux-g…

秋招打卡011(20230807)

文章目录 前言一、今天学习了什么&#xff1f;二、算法----》单调栈1、介绍2、题目 总结 前言 提示&#xff1a;这里为每天自己的学习内容心情总结&#xff1b; Learn By Doing&#xff0c;Now or Never&#xff0c;Writing is organized thinking. 今天拿到了上周面试的结果…

整理mongodb文档:集合名字有类似-等特殊字符串如何处理?

个人博客 整理mongodb文档:集合名字有类似-等特殊字符串如何处理&#xff1f; 首先&#xff0c;先创建一个collection的名字为’collection-test’&#xff0c;这个表名中&#xff0c;colletion的名字有一个特殊字符串“-”&#xff0c;但是后面插入数据的时候会有这么一个错…

SpringCloud(29):Nacos简介

1 什么是配置中心 1.1 什么是配置 应用程序在启动和运行的时候往往需要读取一些配置信息&#xff0c;配置基本上伴随着应用程序的整个生命周期&#xff0c;比如&#xff1a;数据库连接参数、启动参数等。 配置主要有以下几个特点&#xff1a; 配置是独立于程序的只读变量 …

瑞芯微RK3568核心板-4G联网测试

​ &#x1f308;引言 RK3568是瑞芯微针对AIOT和工业市场推出的一款高性能、低功耗、功能丰富的应用处理器。它采用了四核ARM架构64位Cortex-A55处理器&#xff0c;主频高达2.0GHz&#xff0c;集成瑞芯微自研1TOPS算力NPU, 同时集成Mali-G52 2EE GPU&#xff0c;支持4K60fps …

【javaSE】 万字带你认识异常

异常目录 异常的概念与体系结构异常的概念算术异常数组越界异常空指针异常 异常的体系结构 异常的分类编译时异常运行时异常 异常的处理防御式编程LBYL&#xff1a;事前防御型EAFP&#xff1a;事后认错型 异常的抛出注意事项 异常的捕获异常声明throws注意事项 try-catch捕获并…

《吐血整理》高级系列教程-吃透Fiddler抓包教程(35)-Fiddler如何抓取微信小程序的包-下篇

1.简介 通过前边和宏哥的学习&#xff0c;我们了解到Android 7.0 之后增加了对第三方证书的限制&#xff0c;抓包工具&#xff08;charles、fiddler等&#xff09;提供的证书都无法通过校验&#xff0c;也就无法抓取HTTPS请求了&#xff0c;对测试工作影响很大。最近更新的微信…

idea导入新项目报异常:Error:java: Compilation failed: internal java compiler error

修改settings中项目对象Java编译配置恢复正常。 标记的两个地方配置修改和项目一样后&#xff0c;正常。

15年前,互联网蓬勃发展时,你是否还记得这几个标志软件?

互联网蓬勃发展了几十年里&#xff0c;创造很多经典&#xff0c;也成就了很多人物。 我们熟知的有&#xff1a;微软的创始人比尔盖茨&#xff0c;亚马逊的创始人杰夫贝索斯&#xff0c;Google的创始人拉里佩奇等。 给各行各业带来了翻天覆地的变化。 你还记得“IE”吗&#xff…

【云原生】kubernetes中容器的资源限制

目录 1 metrics-server 2 指定内存请求和限制 3 指定 CPU 请求和限制 资源限制 在k8s中对于容器资源限制主要分为以下两类: 内存资源限制: 内存请求&#xff08;request&#xff09;和内存限制&#xff08;limit&#xff09;分配给一个容器。 我们保障容器拥有它请求数量的…

【C++】做一个飞机空战小游戏(六)——给两架飞机设置不同颜色(cout输出彩色字符、结构体使用技巧)

[导读]本系列博文内容链接如下&#xff1a; 【C】做一个飞机空战小游戏(一)——使用getch()函数获得键盘码值 【C】做一个飞机空战小游戏(二)——利用getch()函数实现键盘控制单个字符移动【C】做一个飞机空战小游戏(三)——getch()函数控制任意造型飞机图标移动 【C】做一个飞…

docker安装neo4j

参考文章&#xff1a; 1、Mac 本地以 docker 方式配置 neo4j_neo4j mac docker_Abandon_first的博客-CSDN博客 2、https://www.cnblogs.com/caoyusang/p/13610408.html 安装的时候&#xff0c;参考了以上文章。遇到了一些问题&#xff0c;记录下自己的安装过程&#xff1a; …

ABP:BackgroundJob/Quartz 结合业务实现定时发送

需求描述&#xff1a;我在使用ABP框架&#xff0c;现在想实现一个定时任务功能&#xff0c;描述&#xff1a;每天八点调用特定接口&#xff0c;调用n次后结束不再调用。相关的数据都来自于一个特定的实体“fuck”。 解决方案&#xff1a;可以使用ABP框架自带的定时任务功能来实…

浅聊Cesium.js 后处理原理

浅聊Cesium.js 后处理原理 使用例子: const stages viewer.scene.postProcessStages;const silhouette Cesium.PostProcessStageLibrary.createSilhouetteStage() silhouette.enabled true; stages.add(silhouette);silhouette.uniforms.color Cesium.Color.LIME;涉及到相…

linux内网穿透应用场景有哪些?快解析有什么用处?

随着网络技术的不断发展&#xff0c;无论是工作上还是在生活中人们对网络的依赖和需求越来越高。Linux内网穿透作为一种创新的解决方案&#xff0c;为我们提供了无限可能。 首先我们了解一下Linux操作系统。Linux是一套免费使用和自由传播的类Unix操作系统&#xff0c;是一个基…

流量卡线上销户全教程,剩余的话费还有可能给你退回来!

线上就可以注销手机卡你知道吗&#xff1f;目前三大运营商都开通了线上销户业务了&#xff0c;而且用不完的话费还可以退回来&#xff0c;建议大家点赞收藏起来&#xff0c;以免有需要的时候找不到了&#xff0c;大家好&#xff0c;我是流量卡葫芦妹。 ​ OK&#xff0c;不多废…