C++对象的内存模型

类是创建对象的模板,不占用内存空间,不存在于编译后的可执行文件中;而对象是实实在在的数据,需要内存来存储。对象被创建时会在栈区或者堆区分配内存。

直观的认识是,如果创建了 10 个对象,就要分别为这 10 个对象的成员变量和成员函数分配内存,如下图所示:

不同对象的成员变量的值可能不同,需要单独分配内存来存储。但是不同对象的成员函数的代码是一样的,上面的内存模型保存了 10 份相同的代码片段,浪费了不少空间,可以将这些代码片段压缩成一份。

事实上编译器也是这样做的,编译器会将成员变量和成员函数分开存储:分别为每个对象的成员变量分配内存,但是所有对象都共享同一段函数代码。如下图所示:

成员变量在堆区或栈区分配内存,成员函数在代码区分配内存。

【示例】使用 sizeof 获取对象所占内存的大小:

     
     
  1. #include <iostream>
  2. using namespace std;
  3.  
  4. class Student{
  5. private:
  6. char *m_name;
  7. int m_age;
  8. float m_score;
  9. public:
  10. void setname(char *name);
  11. void setage(int age);
  12. void setscore(float score);
  13. void show();
  14. };
  15.  
  16. void Student::setname(char *name){
  17. m_name = name;
  18. }
  19. void Student::setage(int age){
  20. m_age = age;
  21. }
  22. void Student::setscore(float score){
  23. m_score = score;
  24. }
  25. void Student::show(){
  26. cout<<m_name<<"的年龄是"<<m_age<<",成绩是"<<m_score<<endl;
  27. }
  28.  
  29. int main(){
  30. //在栈上创建对象
  31. Student stu;
  32. cout<<sizeof(stu)<<endl;
  33. //在堆上创建对象
  34. Student *pstu = new Student();
  35. cout<<sizeof(*pstu)<<endl;
  36. //类的大小
  37. cout<<sizeof(Student)<<endl;
  38.  
  39. return 0;
  40. }
#include <iostream>
using namespace std;

class Student{
private:
char *m_name;
int m_age;
float m_score;
public:
void setname(char *name);
void setage(int age);
void setscore(float score);
void show();
};

void Student::setname(char *name){
m_name = name;
}
void Student::setage(int age){
m_age = age;
}
void Student::setscore(float score){
m_score = score;
}
void Student::show(){
cout<<m_name<<“的年龄是”<<m_age<<",成绩是"<<m_score<<endl;
}

int main(){
//在栈上创建对象
Student stu;
cout<<sizeof(stu)<<endl;
//在堆上创建对象
Student *pstu = new Student();
cout<<sizeof(*pstu)<<endl;
//类的大小
cout<<sizeof(Student)<<endl;

return 0;

}
运行结果:

12

12

12



Student 类包含三个成员变量,它们的类型分别是 char *、int、float,都占用 4 个字节的内存,加起来共占用 12 个字节的内存。通过 sizeof 求得的结果等于 12,恰好说明对象所占用的内存仅仅包含了成员变量。



类可以看做是一种复杂的数据类型,也可以使用 sizeof 求得该类型的大小。从运行结果可以看出,在计算类这种类型的大小时,只计算了成员变量的大小,并没有把成员函数也包含在内。



对象的大小只受成员变量的影响,和成员函数没有关系。



假设 stu 的起始地址为 0X1000,那么该对象的内存分布如下图所示:





m_name、m_age、m_score 按照声明的顺序依次排列,和结构体非常类似,也会有内存对齐的问题。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值