静态成员函数_C++:30 C++类成员,成员函数的内存布局

前面两篇文章我相信大家反复读了之后对这节不陌生了:

首先来看代码:

class Demo{public:  //静态成员变量  static const int sx = 0;  //静态函数  static void SF1(){  }public:  //成员变量  int x;public:  //成员函数  void F1(){    cout << "I'm from Demo::F1()" << endl;  }  void F2(){    cout << "I'm from Demo::F1()" << endl;  }  virtual void F3(){    cout << "virtual F3()" << endl;  }public:  //构造函数,C++语法不允许获取构造函数和析构函数地址,要分析其地址,只能查看生产的汇编代码了。  Demo()  {  }  //析构函数  ~Demo()  {  }}; typedef void (Demo::*Func)(); typedef void(*func)();union{  Func f;  void *addr;}ut;int main(int argc, char** argv){  int i = 0;  cout << "main()函数的地址是       :" << std::hex << std::showbase << main << endl;  ut.f = &Demo::F1;  cout << "成员函数F1()的地址是     :" << std::hex << std::showbase << ut.addr << endl;  ut.f = &Demo::F2;  cout << "成员函数F2()的地址是     :" << std::hex << std::showbase << ut.addr << endl;  cout << "静态成员函数SF1()的地址是:" << std::hex << std::showbase << Demo::SF1 << endl;  cout << "静态成员变量sx的地址是   :" << std::hex << std::showbase << &Demo::sx << endl;  cout << "Demo类型实例的大小    :" << sizeof(Demo) << endl;  Demo* pObj = new Demo();  cout << "对象指针变量的地址是     :" << std::hex << std::showbase << &pObj << endl;  cout << "新建对象的地址是         :" << std::hex << std::showbase << pObj << endl;  cout << "成员变量的地址是         :" << std::hex << std::showbase << &pObj->x << endl;  cout << "虚函数表的入口地址     :" << std::hex << std::showbase << pObj << endl;  cout << "虚函数表F3的地址:"<< (int*)*(int*)(pObj) <<endl;     return 0;}

这段代码的运行结果如下:

89171e03b0068b23a213b6148a024dae.png

这里要指出的是大家可以看到静态成员函数和静态成员变量sx的地址都是0x00007FF开头的,实际上他们都在全局数据区域存储(全局变量,静态变量),如果你有耐心,可以断点查看下栈空间内的局部变量i的地址:

caab9756af2621d8c18dc9901aa54af4.png

你会发现栈空间的地址和全局数据区的地址都不一样,这样你也理解了虚函数表的空间。

这样看这张图,你就知道哪些变量在哪里存储了。

c63e162ffb1d67cec49ac730d2640288.png

如果我们修改对象构造的方式,通过在栈上构造一个对象,

int main(int argc, char** argv){   Demo pObj;   cout << "新建对象的地址是         :" << std::hex << std::showbase << &pObj << endl;  cout << "成员变量的地址是         :" << std::hex << std::showbase << &pObj.x << endl;  cout << "静态成员函数SF1()的地址是:" << std::hex << std::showbase << Demo::SF1 << endl;  cout << "静态成员变量sx的地址是   :" << std::hex << std::showbase << &Demo::sx << endl;  ut.f = &Demo::F1;  cout << "成员函数F1()的地址是     :" << std::hex << std::showbase << ut.addr << endl;  ut.f = &Demo::F2;  cout << "成员函数F2()的地址是     :" << std::hex << std::showbase << ut.addr << endl;  cout << "虚函数表的入口地址     :" << std::hex << std::showbase << &pObj << endl;  cout << "虚函数表F3的地址:"<< (int*)*(int*)(&pObj) <<endl;     return 0;}

9006748438cb9053413fda995a06c028.png

我想你现在根据刚才打印的成员变量,成员函数,虚函数表地址已经可以自己画出类成员的内存布局了。

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值