【类和对象】类的作用域 | 类的实例化 | 类对象模型 | this指针

目录

5.类的作用域

6.类的实例化

6.1成员的声明和定义 

6.2实例化出的对象大小

7.类对象模型❗❗

7.1如何计算类对象的大小

7.2类对象的存储方式猜测

7.3结构体内存对齐规则

7.3.1内存对齐

7.3.2大小端  

8.this指针

8.1this指针的引出

8.2this指针的特性

C和C++实现栈的对比


  • 类,对象,成员变量,变量,成员函数的关系

5.类的作用域

类定义了一个新的作用域,类的所有成员都在类的作用域中。

  • 类域能解决一定的命名冲突问题
  • 在类体外定义成员时,需要使用 :: 作用域操作符指明成员属于哪个类域。
//声明.h
class Person
{
public:void PrintPersonInfo();
private:char _name[20];char _gender[3];int  _age;
};//定义.cpp
// 这里需要指定PrintPersonInfo是属于Person这个类域
void Person::PrintPersonInfo()
{cout << _name << " " << _gender << " " << _age << endl;
}

6.类的实例化

用类类型创建对象的过程,称为类的实例化。

  • 类是对对象进行描述的,是一个模型一样的东西,限定了类有哪些成员。
  • 定义出一个类并没有分配实际的内存空间来存储它;比如:入学时填写的学生信息表,表格就可以看成是一个类,来描述具体学生信息。
  •  一个类可以实例化出多个对象,实例化出的对象 占用实际的物理空间,存储类成员变量
  • 类是没有空间的,只有类的实例化出的对象才有具体的空间。

举例:

  • 类就像谜语一样,对谜底来进行描述,谜底就是谜语的一个实例。
  • 谜语:"年纪不大,胡子一把,主人来了,就喊妈妈" 谜底:山羊

举例:做个比方。类实例化出对象就像现实中使用建筑设计图建造出房子,类就像是设计图,只设计出需要什么东西,但是并没有实体的建筑存在,同样类也只是一个设计,实例化出的对象才能实际存储数据,占用物理空间。

 

6.1成员的声明和定义 

  • 无论是成员变量/成员函数在类中,只要还未被实例化成对象,那么它们都是一种声明,因为它们只是类,并不存在空间。
  • 定义并不是给值,而是为类开辟空间,也就是实例化。
  • 定义的时候可以初始化,但是只初始化并不代表定义。
#include<stdlib.h>
#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 s;//定义了s.Init(1,1,1);
❌	Date::_year++;//不能因为它只是声明,并没有开辟空间
❌  Date._month = 100;  // 编译失败:error C2059: 语法错误:“.” return 0;
}

6.2实例化出的对象大小

 移步☞类对象的存储方式。

7.类对象模型❗❗

《深度探索C++对象模型》这本书中对对象模型的描述如下:

有两个概念可以解释C++对象模型:

  • 语言中直接支持面向对象程序设计的部分。
  • 对于各种支持的底层实现机制。

对象模型是指在面向对象编程中,为了描述和处理现实世界中的实体、事物或概念所构建的抽象模型。它通常由类、对象、属性、方法等元素组成,用于描述实体之间的关系、结构和行为。


❓❓问题:类中既可以有成员变量,又可以有成员函数,那么一个类的对象中包含了什么?类对象的存储方式?如何计算一个类的大小? 下面我们来一起探讨。

7.1如何计算类对象的大小

  • 类的实例化=对象
  • sizeof(类名)
  • sizeof(对象)
  • 二者计算都是类实例化的大小,也就是类对象的大小。
  • 一个类的大小,实际就是该类中”成员变量”之和,当然要注意内存对齐
  • 注意空类的大小,空类比较特殊,编译器给了空类一个字节来唯一标识这个类的对象
  • 没有成员变量的类对象大小1字节,标识对象实例化,定义出来存在。
class A
{
public:void PrintA(){cout << _a << endl;}
private:int _a;char _i;
};

❓计算下面类对象的大小: sizeof(A1) : ______ sizeof(A2) : ______ sizeof(A3) : ______

#include<stdlib.h>
#include<iostream>
using namespace std;
// 类中既有成员变量,又有成员函数
//内存对齐存储
class A1 {
public:void f1() {}
private:int _a;char _i;
};// 类中仅有成员函数
//标识对象实例化,定义出来存在
class A2 {
public:void f2() {}
};// 类中什么都没有---空类
class A3
{};
int main()
{cout << sizeof(A1) << endl;cout << sizeof(A2) << endl;cout << sizeof(A3) << endl;return 0;
}

7.2类对象的存储方式猜测

类对象的存储方式有两种:

  • 对象中包含类的各个成员(不用)
  • 代码只保存一份,在对象中保存存放代码的地址(以后)
  • 只保存成员变量,成员函数存放在公共的代码段(现阶段)
//现阶段
class Date
{
public:void Init(int year, int month, int day){_year = year;_month = month;_day = day;}int _year;int _month;int _day;
};int main()
{Date d1;Date d2;//不是一块空间,成员变量各自是各自的空间d1._year;d2._month;//是一块空间,成员函数是一块空间d1.Init(1,1,1);d2.Init(1,1,1);return 0;
}

 【对象中包含类的各个成员】

缺陷:每个对象中成员变量是不同的,但是调用同一份函数,如果按照此种方式存储,当一个类创建多个对象时,每个对象中都会保存一份代码(成员函数的地址),相同代码保存多次,浪费空间。那么如何解决呢?


 【代码只保存一份,在对象中保存存放代码的地址】


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

7.3结构体内存对齐规则

1. 第一个成员在与结构体偏移量为0的地址处。
2. 其他成员变量要对齐到某个数字(对齐数)的整数倍的地址处。

  • 注意:对齐数 = 编译器默认的一个对齐数 与 该成员大小的较小值。
  • VS中默认的对齐数为8。

3. 结构体总大小为:最大对齐数(所有变量类型最大者与默认对齐参数取最小)的整数倍。
4. 如果嵌套了结构体的情况,嵌套的结构体对齐到自己的最大对齐数的整数倍处,结构体的整体大小就是所有最大对齐数(含嵌套结构体的对齐数)的整数倍。


【面试题】

  • 1. 结构体怎么对齐? 为什么要进行内存对齐?
  • 2. 如何让结构体按照指定的对齐参数进行对齐?能否按照3、4、5即任意字节对齐?
  • 3. 什么是大小端?如何测试某机器大端还是小端,有没有遇到过要考虑大小端的场景。
  • 前面C语言进阶我们也讲解过(结构体)C语言之自定义类型_结构体篇(1)-CSDN博客

7.3.1内存对齐

  • 怎样内存对齐?
  • 类对象的包含了什么,需要包括成员函数吗?
  • 为什么要内存对齐?
  • 可以改变对齐参数吗?

【怎么内存对齐---按照规则】 

类的对象存储空间只包含了成员变量,成员函数在公共区域。 

#include<iostream>
using namespace std;
struct A
{
private:int _a;char _i;
};struct B
{
private:char _i;int _a;
};

 【为什么要内存对齐】

  • 对齐是为了提高效率!
  • 硬件规定CPU一次只能读取4/8个字节。
  • 32/64个线,32/64个位,组成0/1的二进制数。(消耗相同,一次只读取32/64位,4/8个字节)
  • 硬件规定:CPU的控制器只能从整数倍开始读取。起始位置必须是整数倍的位置。
  • 以上规定从而导致了:对齐和不对齐读取存在效率上的问题。
  • 对齐读取的次数比不对齐读取的次数多。

【对齐可以改变编译器默认对齐数的大小】

#include<iostream>
using namespace std;//改变默认对齐数的大小
#pragma pack(1)//相当于不对齐struct A
{
private:int _a;char _i;
};struct B
{
private:char _i;int _a;
};int main()
{cout << sizeof(A) << endl;cout << sizeof(B) << endl;return 0;
}

7.3.2大小端  

8.this指针

8.1this指针的引出

8.2this指针的特性

C和C++实现栈的对比

【C语言】

可以看到,在用C语言实现时,Stack相关操作函数有以下共性:

  • 每个函数的第一个参数都是Stack*
  • 函数中必须要对第一个参数检测,因为该参数可能会为NULL
  • 函数中都是通过Stack*参数操作栈的
  • 调用时必须传递Stack结构体变量的地址

结构体中只能定义存放数据的结构,操作数据的方法不能放在结构体中,即数据和操作数据方式是分离开,而且实现上相当复杂一点,涉及到大量指针操作,稍不注意可能就会出错。 


【C++】感谢祖师爷

  • C++中通过类可以将数据 以及 操作数据的方法进行完美结合。
  • 通过访问权限可以控制那些方法在类外可以被调用,即封装。
  • 在使用时就像使用自己的成员一样,更符合人类对一件事物的认知。
  • 而且每个方法不需要传递Stack*的参数了。
  • 编译器编译之后该参数会自动还原,即C++中 Stack *参数是编译器维护的,C语言中需用用户自己维护。

【C语言实现】

#include<stdio.h>
#include<stdlib.h>
#include<assert.h>typedef int DataType;
typedef struct Stack
{DataType* array;int capacity;int size;
}Stack;
void StackInit(Stack* ps)
{assert(ps);ps->array = (DataType*)malloc(sizeof(DataType) * 3);if (NULL == ps->array){assert(0);return;}ps->capacity = 3;ps->size = 0;
}
void StackDestroy(Stack* ps)
{assert(ps);if (ps->array){free(ps->array);ps->array = NULL;ps->capacity = 0;ps->size = 0;}
}
void CheckCapacity(Stack* ps)
{if (ps->size == ps->capacity){int newcapacity = ps->capacity * 2;DataType* temp = (DataType*)realloc(ps->array,newcapacity * sizeof(DataType));if (temp == NULL){perror("realloc申请空间失败!!!");return;}ps->array = temp;ps->capacity = newcapacity;}
}
void StackPush(Stack* ps, DataType data)
{assert(ps);CheckCapacity(ps);ps->array[ps->size] = data;ps->size++;
}
int StackEmpty(Stack* ps)
{assert(ps);return 0 == ps->size;
}
void StackPop(Stack* ps)
{if (StackEmpty(ps))return;ps->size--;
}
DataType StackTop(Stack* ps)
{assert(!StackEmpty(ps));return ps->array[ps->size - 1];
}
int StackSize(Stack* ps)
{assert(ps);return ps->size;
}
int main()
{Stack s;StackInit(&s);StackPush(&s, 1);StackPush(&s, 2);StackPush(&s, 3);StackPush(&s, 4);printf("%d\n", StackTop(&s));printf("%d\n", StackSize(&s));StackPop(&s);StackPop(&s);printf("%d\n", StackTop(&s));printf("%d\n", StackSize(&s));StackDestroy(&s);return 0;
}

【C++实现】 

#include<iostream>
using namespace std;typedef int DataType;
class Stack
{
public:void Init(){_array = (DataType*)malloc(sizeof(DataType) * 3);if (NULL == _array){perror("malloc申请空间失败!!!");return;}_capacity = 3;_size = 0;}void Push(DataType data){CheckCapacity();_array[_size] = data;_size++;}void Pop(){if (Empty())return;_size--;}DataType Top() { return _array[_size - 1]; }int Empty() { return 0 == _size; }int Size() { return _size; }void Destroy(){if (_array){free(_array);_array = NULL;_capacity = 0;_size = 0;}}
private:void CheckCapacity(){if (_size == _capacity){int newcapacity = _capacity * 2;DataType* temp = (DataType*)realloc(_array, newcapacity *sizeof(DataType));if (temp == NULL){perror("realloc申请空间失败!!!");return;}_array = temp;_capacity = newcapacity;}}
private:DataType* _array;int _capacity;int _size;
};
int main()
{Stack s;s.Init();s.Push(1);s.Push(2);s.Push(3);s.Push(4);printf("%d\n", s.Top());printf("%d\n", s.Size());s.Pop();s.Pop();printf("%d\n", s.Top());printf("%d\n", s.Size());s.Destroy();return 0;
}

🙂感谢大家的阅读,若有错误和不足,欢迎指正。

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

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

相关文章

MySQL—redo log、undo log以及MVCC

MySQL—redo log、undo log以及MVCC 首先回忆一下MySQL事务的四大特性&#xff1a;ACID&#xff0c;即原子性、一致性、隔离性和持久性。其中原子性、一致性、持久性实际上是由InnoDB中的两份日志保证的&#xff0c;一份是redo log日志&#xff0c;一份是undo log日志&#xff…

了解常用测试模型 -- V模型、W模型

目录 V模型 测试流程 特点 优、缺点 w模型/双v模型 测试流程 特点 优、缺点 V模型 测试流程 用户需求&#xff1a;产品经理将用户需求转变为软件需求 需求分析与系统设计&#xff1a;验证需求是否正确&#xff0c;确定编程语言和框架 概要设计&#xff1a;项目结构设…

Java数据结构-优先级队列

文章目录 前言一、优先级队列1.1 概念 二、优先级队列的模拟实现2.1 堆的概念2.2 堆的存储方式2.3 堆的创建2.3.1 堆向下调整2.3.2 堆的创建2.3.3 建堆的时间复杂度 2.4 堆的插入与删除2.4.1 堆的插入2.4.2 堆的删除 2.5 用堆模拟实现优先级队列 三、常用接口介绍3.1 PriorityQ…

鼓楼夜市管理wpf+sqlserver

鼓楼夜市管理系统wpfsqlserver 下载地址:鼓楼夜市管理系统wpfsqlserver 说明文档 运行前附加数据库.mdf&#xff08;或sql生成数据库&#xff09; 主要技术&#xff1a; 基于C#wpf架构和sql server数据库 功能模块&#xff1a; 登录注册 鼓楼夜市管理系统主界面所有店铺信…

C++类与对象二

目录 一、类的嵌套 二、对象引用私有数据成员 通过公有函数为私有成员赋值 利用指针访问私有数据成员 利用函数访问私有数据成员 利用引用访问私有数据成员 三、成员函数重载 四、this指针 一、类的嵌套 #include <iostream> using namespace std;class CC1 { p…

华为配置中心AP内漫游实验

华为配置中心AP内漫游示例 组网图形 图1 配置中心AP内漫游组网图 配置流程组网需求配置思路数据规划配置注意事项操作步骤配置文件 配置流程 WLAN不同的特性和功能需要在不同类型的模板下进行配置和维护&#xff0c;这些模板统称为WLAN模板&#xff0c;如域管理模板、射频模…

Spring Cloud Gateway针对指定接口做响应超时时间限制

背景&#xff1a;我做的这个服务中存在要对大数据量做自定义统计的接口和大文件上传接口&#xff0c;接口响应用时会超过gateWay配置的全局用时&#xff0c;如果调整网关全局的超时时间和服务的全局超时时间是不合理的&#xff0c;故此想能否单独针对某个接口进行细粒度超时限制…

林木园区改造VR仿真培训课件提高人们的专业素质

森林经营VR模拟体验摆脱了传统森林经营周期长、实践难及耗材大等问题&#xff0c;借助VR虚拟仿真技术为人们提供一种全新的、沉浸式的森林经营体验&#xff0c;让人们更好地了解森林经营的全周期。 提高人们的环保意识 通过亲身参与森林经营的过程&#xff0c;人们可以更直观地…

4、鸿蒙学习-@ohos.promptAction (弹窗)

创建并显示文本提示框、对话框和操作菜单。 说明 本模块首批接口从API version 9开始支持。后续版本的新增接口&#xff0c;采用上角标单独标记接口的起始版本。 该模块不支持在UIAbility的文件声明处使用&#xff0c;即不能在UIAbility的生命周期中调用&#xff0c;需要在创建…

关于python中数据分析的一些函数

首先先下载numpy函数库 如果使用的pycharm软件&#xff0c;可在设置中下载&#xff0c;在Python interpreter设置里&#xff0c;点击号&#xff0c;搜索numpy点击下载即可 第一部分 1.array()函数 这是一个将类似数组的数据转为数组的函数&#xff0c;我们还可以控制其数组的…

【Unity】程序创建Mesh(二)MeshRenderer、光照、Probes探针、UV信息、法线信息

文章目录 接上文MeshRenderer&#xff08;网格渲染器&#xff09;Materials&#xff08;材质&#xff09;Material和Mesh对应Lighting光照Lightmapping材质中的光照 光源类型阴影全局光照Probes&#xff08;探针&#xff09;Ray Tracing&#xff08;光线追踪&#xff09;Additi…

【C++】map和set深度讲解

> 作者简介&#xff1a;დ旧言~&#xff0c;目前大二&#xff0c;现在学习Java&#xff0c;c&#xff0c;c&#xff0c;Python等 > 座右铭&#xff1a;松树千年终是朽&#xff0c;槿花一日自为荣。 > 目标&#xff1a;熟练掌握map和set容器。 > 毒鸡汤&#xff1a;…

升入理解计算机系统学习笔记

磁盘存储 磁盘是广为应用的保存大量数据的存储设备&#xff0c;存储数据的数量级可以达到几百到几千千兆字节&#xff0c;而基于RAM的存储器只能有几百或几千兆字节。不过&#xff0c;从磁盘上读信息的时间为毫秒级&#xff0c;比从DRAM读慢了10万倍&#xff0c;比从SRAM读慢了…

NodeJs利用腾讯云实现手机发送验证码

本文介绍如何在nodejs实现短信发送&#xff0c;以腾讯云的短信验证为例。 腾讯云中准备工作 首先需要腾讯云的个人或者企业认证的账号&#xff0c;个人会赠送一百条&#xff0c;企业赠送一千条&#xff0c;可以用于测试&#xff0c;地址&#xff1a;腾讯云短信服务。然后需要…

《Ubuntu20.04环境下的ROS进阶学习5》

一、Hector_Mapping构建二维地图 在前面我们已经介绍了如何使用激光雷达来扫描地图&#xff0c;如何用激光雷达来建造地图&#xff0c;本节我们将两者结合起来&#xff0c;通过Hector_Mapping功能包实现SLAM。 二、在仿真环境中进行2D SLAM 1、下载Hector_Mapping sudo apt i…

【论文阅读笔记】Attention Is All You Need

1.论文介绍 Attention Is All You Need 2017年 NIPS transformer 开山之作 回顾一下经典&#xff0c;学不明白了 Paper Code 2. 摘要 显性序列转导模型基于包括编码器和解码器的复杂递归或卷积神经网络。性能最好的模型还通过注意力机制连接编码器和解码器。我们提出了一个新…

【Numpy】练习题100道(26-50题)

#学习笔记# 在学习神经网络的过程中发现对numpy的操作不是非常熟悉&#xff0c;遂找到了Numpy 100题。 Git-hub链接 1.题目列表 26. 下面的脚本输出什么&#xff1f;(★☆☆) print(sum(range(5),-1)) from numpy import * print(sum(range(5),-1)) 27. 考虑一个整数向量…

怎样提升小程序日活?签到抽奖可行吗?

一、 日活运营策略 小程序应该是即用即走的&#xff0c;每个小程序都在用户中有自己的独特定位&#xff0c;可能是生活日常必备&#xff08;美食、团购、商城&#xff09;&#xff0c;也可能是工作办公必备&#xff08;文档、打卡、工具&#xff09;。 如果你想要让自己的小程…

云计算与APP开发,如何利用云端服务提升应用性能?

随着移动应用程序&#xff08;APP&#xff09;的普及&#xff0c;如何提升应用性能成为了开发者们关注的重点之一。而云计算技术的发展为APP开发者提供了全新的解决方案。本文将探讨云计算与APP开发的结合&#xff0c;以及我们公司提出的解决方案&#xff0c;帮助开发者利用云端…

KMP 算法介绍

1. KMP 算法介绍 KMP 算法&#xff1a;全称叫做 「Knuth Morris Pratt 算法」&#xff0c;是由它的三位发明者 Donald Knuth、James H. Morris、 Vaughan Pratt 的名字来命名的。KMP 算法是他们三人在 1977 年联合发表的。 KMP 算法思想&#xff1a;对于给定文本串 T 与模式串 …