【C++】类和对象 - 上

目录

  • 1. 面向过程和面向对象初步认识
  • 2. 类的引入
  • 3. 类的定义
  • 4. 类的访问限定符及封装
    • 4.1 访问限定符
    • 4.2 封装
  • 5. 类的作用域
  • 6. 类的实例化
  • 7. 类对象模型
    • 7.1 如何计算类的大小
    • 7.2 类对象的存储方式猜测
    • 7.3 结构体内存对齐规则
  • 8. this指针
    • 8.1 引出
    • 8.2 this指针的特性
  • 总结

1. 面向过程和面向对象初步认识

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

比如说洗衣服,一般分为以下几个步骤:
在这里插入图片描述
以上的每一个步骤都是需要人来做的,也就是说人是要对每个步骤应该做的事情都要清楚,在语言层面对应依次设计和实现函数来解决遇到的每个问题,这就是面向过程的大体思想。

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

同样是洗衣服,用面向对象的思路则是要把解决的问题分为几个对象:人、衣服、洗衣机,洗衣粉。
洗衣服的过程则为:人把衣服放进洗衣机–>倒入洗衣粉–>启动洗衣机–>最后洗衣机洗完会把衣服甩干。
这里的整个过程主要是这几个对象之间互相交互完成的,而有些步骤,人是不需要知道的,比如洗衣机是如何洗衣服以及如何甩干的,这是面向对象的思想。

2. 类的引入

C语言结构体中只能定义变量,在C++中,结构体内不仅可以定义变量,也可以定义函数。因为C++把结构体升级成了类:

//C语言只能这么玩
struct stu 
{//成员变量char name[20];int age;//...
};//C++
struct stu 
{//成员函数char* getName() {return name;}int getAge() {return age;}//...//成员变量char name[20];int age;//...
};

同样定义一个结构体对象也多了一种方式:

int main()
{//C语言只能像下面这样定义结构体变量struct stu s1;//C++不仅可以支持C语言的写法//下面这种写法也是支持的stu s2;//stu是类名,直接定义变量return 0;
}

C++在语法层面上是完全兼容C的,虽然上面的类型是结构体但是编译器已经把它识别成了类,所以C++更喜欢用关键字class来代替struct

在C++中,用类或者结构定义的变量一般都称之为对象。

3. 类的定义

class className
{// 类体:由成员函数和成员变量组成
};  // 一定要注意后面的分号

class为定义类的关键字,ClassName为类的名字,{}中为类的主体,注意类定义结束时后面分号不能省略。

与定义结构体的语法是一样的。

类体中内容称为类的成员:类中的变量称为类的属性或成员变量; 类中的函数称为类的方法或者成员函数

类的两种定义方式:

  1. 声明和定义全部放在类体中,需注意:成员函数如果在类中定义,编译器可能会将其当成内联函数处理。
  2. 类声明放在.h文件中,成员函数定义放在.cpp文件中,注意:如果定义与声明分离,那么在定义处成员函数名前需要加 类名::,否则编译器无法识别是成员函数还是普通全局函数,如下:
class stu 
{char* getName();int getAge();char name[20];int age;//...
};
char* stu::getName() {return name;
}
int stu::getAge() {return age;
}

若类中定义的成员函数过多,尽量使用第二种方法,这样有利于提高代码的可读和可维护性。

成员变量命名规则建议:

class Date
{
public:void Init(int year){// 这里的year到底是成员变量,还是函数形参?year = year;}
private:int year;
};

为了避免上述不必要的争议,建议把成员变量的名称都加上一些前后缀来区分,比如:

class Date
{
public:void Init(int year){_year = year;}
private://前缀加上一个下划线int _year;//或则int Year;
};

其他方式也可以的,主要目的是为了区分。

4. 类的访问限定符及封装

4.1 访问限定符

struct改为class后,下面代码编译后会报错:
.cn/236d96a7c0034203996011d74454045c.png)

在这里插入图片描述
原因是C++在类中新增了三个访问限定符,分别是:
在这里插入图片描述
说明如下:

  1. public修饰的成员在类外可以直接被访问
  2. protectedprivate修饰的成员在类外不能直接被访问(它俩有些区别,但此处protectedprivate的作用是类似的)
  3. 访问权限作用域从该访问限定符出现的位置开始直到下一个访问限定符出现时为止
  4. 如果后面没有访问限定符,作用域就到 } 即类结束。
  5. class的默认访问权限为private,struct为public(因为struct要兼容C)

从最后一条可以得知报错的原因是:class定义的类,类中的成员变量和成员函数的默认访问权限是private,即外部无法直接访问类中的任何成员,而struct可以是因为它的默认访问权限是public

注意:访问限定符只在编译时有用,当数据映射到内存后,没有任何访问限定符上的区别

为了使得该段代码能够正常通过编译,一种做法是在前面加上public访问限定符:

class stu {
public:char* getName();int getAge();char name[20];int age;
};

这样便可顺利通过编译,但是出于安全性考虑,大部分情况是类中的成员变量不想让外部直接被访问到,但是又需要使用这些变量,因此为了满足这种情况,只需要把成员函数公有即可,成员变量进行私有保护:

class stu {
public:char* getName();int getAge();
private:char name[20];int age;
};

这种做法很好的保护了成员变量被非法访问,若要访问,必须通过对外提供的接口来安全访问这些私有成员。

你要访问使用我,可以,但是你必须要按照我给你提供的方法来操作

问题:C++中struct和class的区别是什么?

答:C++需要兼容C语言,所以C++中struct可以当成结构体使用。另外C++中struct还可以用来定义类。和class定义类是一样的,区别是struct定义的类默认访问权限是public,class定义的类默认访问权限是private。

注意:在继承和模板参数列表位置,struct和class也有区别,后序会介绍

4.2 封装

封装是面向对象的三大特性之一,另外两个分别为:继承和多态。

另外两个后续会介绍

封装不仅在语言中很常见在现实生活中也是如此,那么什么是封装?
封装:将数据和操作数据的方法进行有机结合,隐藏对象的属性或者一些方法的实现细节,仅对外公开接口来和对象进行交互。

举个现实中例子:
计算机的组成以及底层的工作原理十分复杂,但是对于使用者而言压根不太需要关心,只需要知道怎么开机、怎么使用鼠标键盘等来操作计算机就够了。
因此厂商在出厂时,在外部套上壳子,把内部的组成和实现细节全部隐藏起来,对外仅仅暴露出电源以及部分io接口,能让用户和计算机交互即可。

不仅方便了用户,也对计算机本体进行了很好的保护

这是现实层面,在语言层面C++的封装是怎么体现的呢?
它是通过类将数据以及操作数据的方法进行有机结合,通过访问权限来隐藏对象内部实现细节,控制哪些方法可以在类外部直接被使用。

因此封装的本质是一种管理,能让用户更方便的使用类。

5. 类的作用域

类定义了一个新的作用域,类的所有成员都在类的作用域中。在类体外定义成员时,需要使用 :: 作用域操作符指明成员属于哪个类域。

在成员函数内部使用一个变量或者函数时,会先在该函数的局部域中查找,若找不到再去类域中去找,还找不到则会去全局域中去找,找不到就报错,有命名空间也不会去命名空间中去找。

同样可以指定去命名空间域中去找,找不到就报错

6. 类的实例化

用类类型创建对象的过程,称为类的实例化或者叫做对象的定义:

class stu {
public:char* getName();int getAge();
private:char name[20];				  int age;
};
int main()
{//对象的实例化stu s1;stu s2;return 0;
}

类类型和结构体类型是一样的,本质只是定义一个新的类型,类型是不占用内存空间的,只有用该类型实例化出一个对象(变量)才会在内存中开辟空间来存储类中声明的那些成员。

就好比建筑图纸,图纸只有一份,并不占地方,只有通过图纸建造出很多实体建筑才会占用地方

在这里插入图片描述

7. 类对象模型

7.1 如何计算类的大小

先说结论:计算一个类的大小与计算一个结构体类型的大小规则是完全一致的。

class stu {
public:char* getName();int getAge();
private:char name[20];				  int age;
};
int main()
{cout << sizeof(stu) << endl;return 0;
}

这里的运行结果为24,字符数组占20个字节,整形占了4个字节,非常奇怪的是类中的成员函数并没有占用空间,这是为什么?

先举个简单的例子:一个小区有多户人家,每家的房子一定都是独立的,都有各自的卧室、客厅和厨房等等,但是若要建一个游泳池或者健身房有没有必要给每一家都建一个呢?是可以的,但是没必要,因为这类建筑对于每户人家的作用是一致的,会造成大量的空间浪费,所以比较好的方法是把它们独立出来在一个公共的区域去建,这样所有人都可以用,很大程度的减少了空间的浪费。

有了这个例子对于上面的问题就比较好理解了,其实类比到类,是一样的。
多户人家可以当作多个实例化出来的对象,每个对象中的成员变量可以当作对应的卧室、客厅和厨房,成员函数则当作健身房,其中成员变量必须是存储在不同的空间中,而因为每个对象中的成员函数的作用是相同的,要是每个对象都存储的话会造成空间浪费,因此在存储的时候可以把它们独立出来放在一块公共的代码段,不同的对象都可以找到然后调用它。

7.2 类对象的存储方式猜测

关于类对象的存储模型有以下三种:

  1. 对象中包含类的各个成员
    在这里插入图片描述
    这种做法是有缺陷的,每个对象中成员变量是不同的,但是调用同一份函数,如果按照此种方式存储,当一个类创建多个对象时,每个对象中都会保存一份代码,相同代码保存多次,浪费空间。

  2. 代码只保存一份,在对象中保存存放函数的地址
    在这里插入图片描述
    很大程度的解决了第一种空间浪费的问题,但没有完全解决。

  3. 只保存成员变量,成员函数存放在公共的代码段
    在这里插入图片描述
    这种是最优的一种实现方法,有了上面提到的例子,可以发现的是类对象的存储模型就是采用的第三种方法,完美解决了空间的浪费。

注意:没有成员变量的类或者空类的大小只有一个字节,作为占位符不存储有效数据,告诉编译器对象存在。

7.3 结构体内存对齐规则

最开始提到过,计算类的大小与计算结构体类型大小的规则是一样的,都要遵循结构体内存对齐原则,具体的在这篇文章中详细地介绍了,这里不再赘述。

8. this指针

8.1 引出

先定义一个日期类:

class Date{
public:void Init(int year, int month, int day) {_year = year;_month = month;_day = day;}void Print(){cout << _year << "-" << _month << "-" << _day << endl;}
private:int _year; // 年int _month;// 月int _day; // 日
};
int main()
{Date d1, d2;d1.Init(2022,1,11);d2.Init(2022, 1, 12);d1.Print();d2.Print();return 0;
}

输出结果:
在这里插入图片描述
在主函数中定义了两个对象,分别调用Print函数输出了不同的日期,而两个对象都是调的同一个函数,上面说过不同的对象它们的成员函数所在的区域则是相同的,函数体中也没有不同对象的区分,那么函数是如何识别是哪个对象调用了我呢?

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

因此上面的代码在被编译器处理后变成下面这样子:

 //所有的非静态成员函数都会被处理,不一一写了//处理前:
void Print(){cout << _year << "-" << _month << "-" << _day << endl;
}
d1.Print();
d2.Print();//处理后:
void Print(Date* const this) {cout << this->_year << "-" << this->_month << "-" << this->_day << endl;
}d1.Print(&d1);d2.Print(&d2);

需要注意的是,this指针传参是编译器的工作,用户不需要显式地传,也不能,否则会报错,但是却可以在函数体里显式地使用this指针,因为有些场景会使用它。

8.2 this指针的特性

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

两个题:

  1. this指针存在哪里?
    this指针虽然是编译器隐式传递的参数,但本质也是一个形式参数,是参数就会在函数调用时被依次压入到栈中,函数调用结束形参也随之被销毁了。
  2. this指针可以为空吗?
    可以,但如果为空,则不可以在函数体中解引用访问对应的成员变量,因为对空指针解引用会报错。

总结

C++中的类是从C语言里的结构体进化而来,并且引入了面向对象的思想,即封装,使得成员变量和成员方法可以一起放在类中,使其之间的联系更加紧密,同时增加了类的访问限定符,不仅使得访问类中的成员更加规范化,而且也屏蔽了底层的部分实现细节,提高了代码的保密性。

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

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

相关文章

打开域名跳转其他网站,官网被黑解决方案(Linux)

某天打开网站&#xff0c;发现进入首页&#xff0c;马上挑战到其他赌博网站。 事不宜迟&#xff0c;不能让客户发现&#xff0c;得马上解决 我的网站跳转到这个域名了 例如网站跳转到 k77.cc 就在你们部署的代码的当前文件夹下面&#xff0c;执行下如下命令 find -type …

Electron 系统通知 Notification 实践指南

系统通知是桌面应用的常见功能&#xff0c;用于给用户发送提醒&#xff08;刷下存在感 &#x1f642;&#xff09;&#xff0c;还能帮定点击事件以便后续的操作。 Electron 自带通知模块&#xff0c;下方代码是一个简单的示例 const { Notification } require(electron)cons…

电脑维护指南:让你的战友始终高效稳定

&#x1f337;&#x1f341; 博主猫头虎 带您 Go to New World.✨&#x1f341; &#x1f984; 博客首页——猫头虎的博客&#x1f390; &#x1f433;《面试题大全专栏》 文章图文并茂&#x1f995;生动形象&#x1f996;简单易学&#xff01;欢迎大家来踩踩~&#x1f33a; &a…

【Ansible】

目录 一、Ansible简介二、ansible 环境安装部署1、管理端安装 ansible 三、ansible 命令行模块&#xff08;重点&#xff09;1&#xff0e;command 模块2&#xff0e;shell 模块3、cron 模块4&#xff0e;user 模块5&#xff0e;group 模块6&#xff0e;copy 模块&#xff08;重…

【论文精读】Self-Attentive Assocative Memory,2020

目录 1 引言2 Outer product attention (OPA)3 Self-attentive Associative Memory (SAM)4 SAM-based Two-Memory Model (STM)4.1 M i M^i Mi写操作4.2 M r M^r Mr读操作4.3 M i M^i Mi读操作和 M r M^r Mr写操作过程4.4 用 M r M^r Mr实现item转移4.5 模型输出 o t o_t ot​…

c语言内存函数的深度解析

本章对 memcpy&#xff0c;memmove&#xff0c;memcmp 三个函数进行详解和模拟实现&#xff1b; 本章重点&#xff1a;3个常见内存函数的使用方法及注意事项并学会模拟实现&#xff1b; 如果您觉得文章不错&#xff0c;期待你的一键三连哦&#xff0c;你的鼓励是我创作的动力…

el-table数据处理

在写表格时遇到&#xff0c;后端返回的数据是对象&#xff0c;并且缺少字段 1.每一条数据加上 一个字段 2.将对象转成数组 以下是数据 {"groupA": {"groupName": null,"orgName": null,"orgId": null,"allPeoper": &quo…

IntersectionObserver实现小程序长列表优化

IntersectionObserver实现小程序长列表优化 关于 IntersectionObserver 思路 这里以一屏数据为单位【一个分页的10条数据&#xff0c;最好大于视口高度】&#xff0c; 监听每一屏数据和视口的相交比例&#xff0c;即用户能不能看到它 只将可视范围的数据渲染到页面上&#x…

Oracle 19c 报ORA-704 ORA-01555故障处理---惜分飞

异常断电导致数据库无法启动,尝试对数据文件进行recover操作,报ORA-00283 ORA-00742 ORA-00312错误,由于redo写丢失无法正常应用 D:\check_db>sqlplus / as sysdba SQL*Plus: Release 19.0.0.0.0 - Production on 星期日 7月 30 07:49:19 2023 Version 19.3.0.0.0 Copyrig…

利用读时建模等数据分析能力,实现网络安全态势感知的落地

摘要&#xff1a;本文提出一种基于鸿鹄数据平台的网络安全态势感知系统&#xff0c;系统借助鸿鹄数据平台读时建模、时序处理、数据搜索等高效灵活的超大数据存储和分析处理能力&#xff0c;支持海量大数据存储、分类、统计到数据分析、关联、预测、判断的网络安全态势感知能力…

CentOS7系统Nvidia Docker容器基于TensorFlow2.12测试GPU

CentOS7系统Nvidia Docker容器基于TensorFlow1.15测试GPU 参考我的另一篇博客 1. 安装NVIDIA-Docker的Tensorflow2.12.0版本 1. 版本依赖对应关系&#xff1a;从源代码构建 | TensorFlow GPU 版本Python 版本编译器构建工具cuDNNCUDAtensorflow-2.6.03.6-3.9GCC 7.3.1Ba…

beego通过gorm访问mysql数据库

一、下载golang 二、解压下载包到C盘 三、配置golang系统环境变量 四、进入新建的工作目录C:\project下载并安装beego 五、将新生成的bee.exe所在的路径c:\project\bin加入到系统变量path里面 六、下载并安装mysql 例如在上图中&#xff0c; 选“No thanks,just start my down…

如何在3ds max中创建可用于真人场景的巨型机器人:第 3 部分

推荐&#xff1a; NSDT场景编辑器助你快速搭建可二次开发的3D应用场景 1. 创建腿部装备 步骤 1 打开 3ds Max。 打开在本教程最后一部分中保存的文件。 打开 3ds Max 步骤 2 转到创建> 系统并单击骨骼。 创建>系统 步骤 3 为的 侧视口中的腿&#xff0c;如下图所示…

Java 程序员:本是为了跳槽刷完 1000 道真题,想不到被老板知道直接给我升职

同事&#xff1a;前阵子听说你要跳槽&#xff0c;现在准备得怎么样啊&#xff1f; 程序员 T&#xff1a;不跳了 同事&#xff1a;啊&#xff1f;为什么&#xff1f; 程序员 T&#xff1a;涨薪了呗&#xff1f; 同事&#xff1a;真的吗&#xff1f;涨了多少&#xff1f;你自…

R语言无法调用stats.dll的问题解决方案[补充]

写在前面 在去年10月份&#xff0c;出过一起关于R语言无法调用stats.dll的问题解决方案,今天&#xff08;你看到后是昨天&#xff09;不知道为什么&#xff0c;安装包&#xff0c;一直安装不了&#xff0c;真的是炸裂了。后面再次把R与Rstuido升级。说实话&#xff0c;我是真不…

C语言指针详解

C语言指针详解 字符指针1.如何定义2.类型和指向的内容3.代码例子 指针数组1.如何定义2.类型和内容 数组指针1.如何定义2.类型和指向类型3.数组名vs&数组名数组指针运用 数组参数&指针参数一维数组传参二维数组传参一级指针传参二级指针传参 函数指针1.如何定义2.类型和…

Java ~ Collection/Executor ~ DelayQueue【总结】

前言 文章 相关系列&#xff1a;《Java ~ Collection【目录】》&#xff08;持续更新&#xff09;相关系列&#xff1a;《Java ~ Executor【目录】》&#xff08;持续更新&#xff09;相关系列&#xff1a;《Java ~ Collection/Executor ~ DelayQueue【源码】》&#xff08;学…

Elasticsearch笔记

迈向光明之路&#xff0c;必定荆棘丛生。 文章目录 一、Elasticsearch概述二、初识ES倒排索引1. 正向索引2. 倒排索引 三、ES环境搭建1. 安装单机版ES2. 安装Kibana3. 安装ik分词器3.1 在线安装ik插件3.2.离线安装ik插件&#xff08;推荐方式&#xff09;3.3 自定义词典 四、ES…

Linux上定位线上CPU飙高

【模拟场景】 写一个java main函数&#xff0c;死循环打印 System.out.println(“111111”) &#xff0c; 将其打成jar包放在linux中执行 1、通过TOP命令找到CPU耗用最厉害的那个进程的PID 2、top -H -p 进程PID 找到进程下的所有线程 可以看到 pid 为 94384的线程耗用cpu …

VUE3-04

1. 编写代码过程中的问题与解决 1.1 错误&#xff1a;cant read property of undefined(name) &#xff08;1&#xff09;首先定位错误的位置 &#xff08;2&#xff09;逐一排查问题&#xff1a;注释代码&#xff1b;debugger&#xff1b;console.log &#xff08;3&#xff0…