【C++深度剖析教程28】C++对象模型分析

学习交流加

  • 个人qq:
    1126137994
  • 个人微信:
    liu1126137994
  • 学习交流资源分享qq群:
    962535112

今天记录学习的内容是:C++对象模型分析!!!

本质分析:
class是一种特殊的struct

  • 在内存中class依旧可以看做是变量的集合
  • class与struct遵循相通的对齐准则
  • class中的成员函数与成员变量是分开存放的
    *每个对象有独立的成员变量
    *所有对象共享类中的成员函数
    下图中的结果是什么?
    这里写图片描述

上面的结果需要分情况讨论,但是不管结果是多少,它们两个都是相等的,因为class本身就是一种特殊的struct。
下面来总结一下,结构体的字节对齐,以及它在不同编译器中的不同的对齐方式!

Linux系统下:GCC默认为4字节对齐,有多个低字节类型的变量(这多个低字节类型(低于字节对齐的大小)的变量是连续存储的,中间没有其他高于4字节类型变量),则这多个低字节类型的变量组成后的字节总数,再与4字节对齐(即4的倍数)
windows下的VS编译器:以最高字节类型的字节数对齐,同理有多个低字节类型的变量的话(这多个低字节类型的变量(低于对齐字节的大小)是连续存储的,中间没有其他类型变量),加在一起再与最大类型字节数做对比(即最大字节数的倍数)

注:
如果是这种情况下:

struct A
{char s;int a;double b;
};

则char s 与int a 合并后的字节总署为5<8,所以char s与int a 合并后再与8字节对齐,所以该结构体的字节数为16,而不是24;
下面看一个例子:

#include <iostream>
#include <string>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 bytescout << "sizeof(a) = " << sizeof(a) << endl;cout << "sizeof(B) = " << sizeof(B) << endl;    // 20 bytesa.print();//定义结构体指针p指向A对象a,B* p = reinterpret_cast<B*>(&a);//通过结构体指针p来操作对象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;
}

运行结果如下:
这里写图片描述

注意:以上是在linux上用gcc编译器进行编译,所以结构体的对齐方式是以4字节的对齐的,所以结果为20字节,如果放到vs上编译,那么该结构体的大小就为24字节。

同时,我们可以看出,我们可以通过将结构体指针指向类对象来直接操作对象,这也说明了,class是一种特殊的结构体!!!

结论:
运行时的对象,退化为结构体的形式:

  1. 所有成员变量在内存中依次排布
  2. 成员变量间可能存在内存空隙
  3. 可以通过内存地址直接访问成员变量
  4. 访问权限关键字在运行时失效
  5. 类中的成员函数位于代码段中
  6. 调用成员函数时,对象地址作为参数隐式传递
  7. 成员函数通过对象地址访问成员变量
  8. C++语法规则隐藏了对象地址的传递过程

下面再看一个例子,来探讨C++中对象的本质:

以下C程序是用来模拟C++中的对象模型的:
50-2.h为:

#ifndef _50_2_H_
#define _50_2_H_typedef void Demo; //定义Demo为一个void类型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

50-2.c为:

#include "50-2.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);
}

main.c为:

#include <stdio.h>
#include "50-2.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;
}

以上c程序,实现了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;
}

总结:

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

想获得各种学习资源以及交流学习的加我:
qq:1126137994
微信:liu1126137994
可以共同交流关于嵌入式Linux,操作系统,C++语言,C语言,数据结构与算法等技术问题。

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

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

相关文章

【C++深度剖析教程29】C++对象模型分析下

加qq1126137994&#xff0c;微信&#xff1a;liu1126137994 一起学习更多技术&#xff01;&#xff01;&#xff01; 今天来继续学习C对象模型&#xff01; 在C编译器内部&#xff0c;类可以理解为结构体子类是由父类成员叠加子类新成员得到的 下面来写一个程序分析C的继承…

java 思维导图

最近有时间&#xff0c;又整理了一下java 思维导图&#xff0c;持续优化中 子模块拆分总结&#xff1a;java基础 java思维导图 总结

【常见笔试面试算法题12续集一】动态规划算法案例1台阶问题练习题

加qq1126137994 一起学习更多技术&#xff01;&#xff01;&#xff01; 以下问题&#xff0c;都可以用非动态规划的方法做&#xff0c;我为了整理动态规划的方法思路&#xff0c;就全部用动态规划的思路来解决问题&#xff0c;这样还可以简化问题的处理&#xff0c;是时间复杂…

linux 思维导图

工作中使用总结的linux 分享一下

【常见笔试面试算法题12续集二】动态规划算法案例2矩阵最小路径和练习题

加qq1126137994 一起学习更多技术&#xff01;&#xff01;&#xff01; 有一个矩阵map&#xff0c;它每个格子有一个权值。从左上角的格子开始每次只能向右或者向下走&#xff0c;最后到达右下角的位置&#xff0c;路径上所有的数字累加起来就是路径和&#xff0c;返回所有的…

mysql数据库思维导图

先前总结的一下数据库的思维导图&#xff0c;和大家分享一下&#xff0c;持续更新&#xff1b; 最近面试数据仓库工程师&#xff0c;过程中&#xff0c;问到索引类型&#xff0c;只是记得&#xff0c;普通索引&#xff0c;唯一索引&#xff0c;还有主键&#xff0c;面试中让说…

【常见笔试面试算法题12续集三】动态规划算法案例分析3 LIS练习题(最长上升子序列)

学习交流加 个人qq&#xff1a; 1126137994个人微信&#xff1a; liu1126137994学习交流资源分享qq群&#xff1a; 962535112 这是一个经典的LIS(即最长上升子序列)问题&#xff0c;请设计一个尽量优的解法求出序列的最长上升子序列的长度。 给定一个序列arr及它的长度n(长度小…

【常见笔试面试算法题12续集四】动态规划算法案例分析4 LCS练习题练习题(最长公共子序列的长度)

学习交流加 个人qq&#xff1a; 1126137994个人微信&#xff1a; liu1126137994学习交流资源分享qq群&#xff1a; 962535112 给定两个字符串A和B&#xff0c;返回两个字符串的最长公共子序列的长度。例如&#xff0c;A"1A2C3D4B56”&#xff0c;B"B1D23CA45B6A”&am…

hive 思维导图

1.整理了word文档&#xff0c;但是感觉没有思维导图清晰 &#xff0c;持续更新中 ​ hive 工作中分享。

结构思考力思维导图

个人提升&#xff0c;喜欢钻研的同时&#xff0c;需要增加一下&#xff0c;其他方面的知识&#xff0c;分享一下&#xff1a;

常用内存分配函数的说明

下面哪种C/C 分配内存的方法会将分配的空间初始化为0 A malloc() B calloc() C realloc() D new[ ] 1) malloc 函数&#xff1a; void *malloc(unsigned int size) 在内存的动态分配区域中分配一个长度为size的连续空间&#xff0c;如果分配成功&#xff0c;则返回所分配…

前端学习(120):css精灵和好处

精灵图的目的&#xff1a;很多大型网站在首次加载的时候需要加载很多小图片&#xff0c;如果在同一时间&#xff0c;服务器又拥堵的话&#xff0c;就会导致页面加载慢。为了解决这一问题&#xff0c;就用了精灵图来解决加载时间过长影响用户体验的问题。 精灵图的本质&#xf…

shell 思维导图

分享一下&#xff1b;

Linux 总线、设备、驱动模型的探究

学习交流加 个人qq&#xff1a;1126137994 个人微信&#xff1a;liu1126137994 学习交流资源分享qq群&#xff1a;962535112之前一直做项目&#xff0c;做项目的过程虽然也学习到了不少知识&#xff0c;但是&#xff0c;一直没有好好研究总线设备驱动的机制&#xff0c;今天来学…

二叉树的层序遍历详细讲解(附完整C++程序)

加qq1126137994 微信liu1126137994 一起学习更多技术 1、原理&#xff1a; 层序遍历所要解决的问题很好理解&#xff0c;就是按二叉树从上到下&#xff0c;从左到右依次打印每个节点中存储的数据。如下图&#xff1a; 按层序遍历的原则&#xff0c;打印顺序依次应该是&…

【C++深度剖析教程30】C++中抽象类和接口

加qq1126137994 微信&#xff1a;liu1126137994 1、面向对象中的抽象概念; 在现实中&#xff0c;需要知道具体的图形类型才能知道如何求面积&#xff0c;但是对于抽象的‘图形’&#xff0c;我们是没法求其面积的&#xff0c;而且对其求面积也是没有意义的&#xff1a; cl…

有效沟通力思维导图

有效沟通&#xff0c;适用于生活和工作中&#xff0c;甚至是孩子教育上面&#xff0c;分享一下&#xff1b;

数据结构思维导图

学习是一个不断渐进的过程&#xff0c;最近整理了一下数据结构思维导图&#xff0c;分享一下&#xff0c;后续更新中 总结&#xff1a;算法实际上属于&#xff0c;数据建模&#xff0c;首先是问题的抽象&#xff0c;采用数学公式来表示&#xff08;数据建模&#xff1a;将问题…

(转)代理模式(Proxy)

原文地址&#xff1a;http://www.cnblogs.com/QinBaoBei/archive/2010/05/18/1737866.html 为了深刻点理解代理模式&#xff0c;我们先来看一个 Demo &#xff0c; 首先这个 Demo 是用来测试 QQ 号码是否在线&#xff0c; 这里涉及到的内容是 Web 服务的使用&#xff0c; 这个 …