【C++基础入门】44.C++中对象模型分析(上)

news/2024/10/18 1:30:08/

一、回归本质

  • class 是一种特殊的 struct
    • 在内存中 class 依旧可以看作变量的集合
    • classstruct 遵循相同的内存对齐规则
    • class 中的成员函数成员变量是分开存放的
      • 每个对象有独立的成员变量
      • 所有对象共享类中的成员函数
  • 值得思考的问题

        下面看一个对象内存布局的代码:

#include <iostream>using namespace std;class A
{int i;int j;char c;double d;
public:void print(){cout << "i = " << i << ","<< "j = " << j << ","<< "c = " << c << ","<< "d = " << d << endl;}
};struct B
{int i;int j;char c;double d;
};int main()
{A a;cout << "sizeof(A) = " << sizeof(A) <<  endl;  //20 bytecout << "sizeof(a) = " << sizeof(a) <<  endl;cout << "sizeof(B) = " << sizeof(B) <<  endl;a.print();B* p = reinterpret_cast<B*>(&a);p->i = 1;p->j = 2;p->c = 'c';p->d = 3;a.print();p->i = 100;p->j = 200;p->c = 'C';p->d = 3.14;a.print();return 0;
}

        输出结果如下:

  1. 首先,学习采用的为 ubuntu 10.10,其 gcc 编译器暂时不支持 8 字节对齐,默认按照 4 字节对齐,所以 A 和 B 中的 d 的对齐参数为 4,故 A 和 B 占用内存大小为 20 字节(详情看:【C语言进阶剖析】24.C语言中的 #pragma 使用分析)
  2. 成员变量可能排布在栈空间、可能排布在堆空间,还有可能排布在全局数据区,而成员函数只可能存在于代码段
  3. 为什么要用 reinterpret_cast 这个关键词,这是因为我们需要解释 a 这个对象代表的内存,关于 C++ 强制类型转换,可以看:【C++基础入门】9.C++中的类型转换
  4. 通过后面两个例子说明一个对象是一个特殊的结构体,由于 a 对象成员变量与 B 结构体中的变量在内存中的排布相同,因此,可以利用指针修改 a 这个对象中的私有成员

二、C++对象模型分析

  • 运行时的对象退化为结构体的形式
    • 所有成员变量在内存中依次排布
    • 成员变量间可能存在内存空隙
    • 可以通过内存地址直接访问成员变量
    • 访问权限关键字在运行时失效
  • 类中的成员函数位于代码段
  • 调用成员函数时对象地址作为参数隐式传递
  • 成员函数通过对象地址访问成员变量
  • C++ 语法规则隐藏了对象地址的传递过程

        下面进行对象本质分析:

#include <iostream>
#include <string>using namespace std;class Demo
{int mi;int mj;
public:Demo(int i, int j){mi = i;mj = j;}int getI(){return mi;}int getJ(){return mj;}int add(int value){return mi + mj + value;}
};int main()
{Demo d(1, 2);cout << "sizeof(d) = " << sizeof(d) << endl;cout << "d.getI() = " << d.getI() << endl;cout << "d.getJ() = " << d.getJ() << endl;cout << "d.add(3) = " << d.add(3) << endl;return 0;
}

        输出结果如下:

        d 对象的地址被传到了 getI() 这个函数的内部,但是这个传递过程在 C++ 的代码中是看不到的,下面我们用 C 语言模拟实现这个过程,挖掘编译器背后的故事 

        main.c

#include <stdio.h>
#include "demo.h"int main()
{Demo* d = Demo_Create(1, 2);             // Demo* d = new Demo(1, 2);printf("d.mi = %d\n", Demo_GetI(d));     // d->getI();printf("d.mj = %d\n", Demo_GetJ(d));     // d->getJ();printf("Add(3) = %d\n", Demo_Add(d, 3));    // d->add(3);// d->mi = 100;Demo_Free(d);return 0;
}

        demo.h

#ifndef _DEMO_H_
#define _DEMO_H_typedef void Demo;Demo* Demo_Create(int i, int j);
int Demo_GetI(Demo* pThis);
int Demo_GetJ(Demo* pThis);
int Demo_Add(Demo* pThis, int value);
void Demo_Free(Demo* pThis);#endif

         demo.c

#include "demo.h"
#include "malloc.h"struct ClassDemo
{int mi;int mj;
};Demo* Demo_Create(int i, int j)
{struct ClassDemo* ret = (struct ClassDemo*)malloc(sizeof(struct ClassDemo));if( ret != NULL ){ret->mi = i;ret->mj = j;}return ret;
}int Demo_GetI(Demo* pThis)
{struct ClassDemo* obj = (struct ClassDemo*)pThis;return obj->mi;
}int Demo_GetJ(Demo* pThis)
{struct ClassDemo* obj = (struct ClassDemo*)pThis;return obj->mj;
}int Demo_Add(Demo* pThis, int value)
{struct ClassDemo* obj = (struct ClassDemo*)pThis;return obj->mi + obj->mj + value;
}void Demo_Free(Demo* pThis)
{free(pThis);
}

         输出结果如下:

  1. 由于Demo是 void 类型,所以 pThis 需要强制类型转化为 struct ClassDemo 类型
  2. main.c 中,d 是指针,指向的就是模拟对象的地址
  3. 运行 d->mi = 100; 会报如下错误,从面向对象的观点来看,mi 是私有的,在类的外部是不能访问的,只能通过成员函数进行访问,这是面向对象中的信息隐藏,由于 C语言中没有 private 关键字,所以只能通过 void 进行信息隐藏,从而模拟 private 关键字

 

三、小结

  • C++ 中的类对象在内存布局上与结构体相同
  • 成员变量成员函数在内存中分开存放
  • 访问权限关键字在运行时失效
  • 调用成员函数时对象地址作为参数隐式传递
     

http://www.ppmy.cn/news/1185664.html

相关文章

LeetCode 面试题 16.08. 整数的英语表示

文章目录 一、题目二、C# 题解 一、题目 给定一个整数&#xff0c;打印该整数的英文描述。 示例 1: 输入: 123 输出: “One Hundred Twenty Three” 示例 2: 输入: 12345 输出: “Twelve Thousand Three Hundred Forty Five” 示例 3: 输入: 1234567 输出: “One Million Two …

openGauss学习笔记-109 openGauss 数据库管理-管理用户及权限-角色

文章目录 openGauss学习笔记-109 openGauss 数据库管理-管理用户及权限-角色109.1 创建、修改和删除角色109.2 内置角色 openGauss学习笔记-109 openGauss 数据库管理-管理用户及权限-角色 角色是一组用户的集合。通过GRANT把角色授予用户后&#xff0c;用户即具有了角色的所有…

由浅入深C系列八:如何高效使用和处理Json格式的数据

如何高效使用和处理JSON格式的数据 问题引入关于CJSON示例代码头文件引用处理数据 问题引入 最近的项目在用c处理后台的数据时&#xff0c;因为好多外部接口都在使用Json格式作为返回的数据结构和数据描述&#xff0c;如何在c中高效使用和处理Json格式的数据就成为了必须要解决…

qml之ui控件

文章目录 ui控件移动版风格嵌套页面并排界面 ui控件 Qt Quick控件用于创建由标准化组件&#xff08;如按钮、标签、滑块等&#xff09;构建的用户界面。 QtQuick.Controls&#xff1a;基本控件。QtQuick.Templates&#xff1a;为控件提供行为化的、非可化视的基本类型。QtQui…

Django 尝试SSE报错 AssertionError: Hop-by-hop headers not allowed 的分析

情况描述 近期计划测试一下django对日志打印的支持&#xff0c;一般都是用websocket的方式&#xff0c;想测试一下SSE (Server-sent events)的服务端推送&#xff0c;发现过程中存在报错&#xff1a; Traceback (most recent call last):File "D:\Software\Anaconda3\li…

Flask后端开发(二) - 功能实现和项目总结

目录 1. 功能1:修改文件参数值1.1. 获取网页端传参1.2. 读取文件1.2.1. 一般文件读取方式1.2.2. 特殊文件 —— mlx文件1.2.3. 特殊文件 —— .xlx文件1.3. 查找数据修改位置,替换数据2. 功能2:读取结果数据2.1. 实时数据展示如何存储相关数据?2.2. 读取相关数据,整理、打…

550MW发电机变压器组继电保护的整定计算及仿真

摘要 电力系统继电保护设计是根据系统接线图及要求选择保护方式&#xff0c;进行整定计算&#xff0c;电力系统继电保护的设计与配置是否合理直接影响到电力系统的安全运行。如果设计与配置不当&#xff0c;保护将不能正确工作&#xff0c;会扩大事故停电范围&#xff0c;造成…

MySQL数据库干货_07—— MySQL中的约束

MySQL中的约束 本专栏从本篇开始正式介绍MySQL中的约束内容&#xff0c;这是关系型数据库的一个重点&#xff0c;在接下来的几篇博文中我会详细介绍每种约束&#xff0c;包括概念&#xff0c;创建方式&#xff0c;应用场景等等&#xff0c;希望小伙伴们关注&#xff01;约束概…