[GeekBand] C++面向对象高级编程-2

对象模型(Object Model)

C++对象的内容分为数据成员(class data members)与成员函数(class member functions),其中数据成员又可分为静态数据成员、非静态数据成员两类,成员函数又可分为静态函数、非静态函数与虚函数三类。

存在虚函数的类对象里连续存放着虚函数表指针(Vptr)与静态数据成员,Vptr 指向虚函数表(Vtbl),Vtbl 存放于其他区域,而Vtbl里连续存放着此类对象的所有虚函数的地址。

  
class Base {

  private:
    int i;
    double d;

  public:
    virtual f( ) {    }
    virtual g( ) {    }

};

可以通过如下代码验证虚函数的相关机制:

定义一种函数指针类型 Fun:typedef void(*Fun)(void);

定义Base类的对象 b:Base b;

虚函数表指针(Vptr)的地址:(int*)(&b)

虚函数表(Vtbl)的地址:*((int*)(&b))

第一个虚函数 f( ) 的地址:(int*)*(int*)(&b) + 0,或:(int*)(*(int*)(&b)) + 0

第二个虚函数 g( ) 的地址: (int*)*(int*)(&b) + 1,或:(int*)(*(int*)(&b)) + 1

将第一个虚函数的地址转化为Fun类型的函数指针:(Fun)*((int*)*(int*)(&b) + 0)

验证程序如下:

// ObjectModel.cpp
#include <iostream>
#include <stdlib.h>

using namespace std;

class Fruit{
    int no = 1;
    double weight = 0.1;
    char key = 'k';
public:
    Fruit(int _no, double _weight, char _key): no(_no), weight(_weight), key(_key) {
        cout << "Calling Fruit(int, double, char)..." << endl;
    }
    Fruit() {
        cout << "Calling Fruit()..." << endl;
    }
    void print() {   }
    virtual void process(){ cout << "Fruit::process()" << endl; }
    virtual ~Fruit() {
        cout << "Calling ~Fruit()..." << endl;
    }
};

class Apple: public Fruit{
    int size = 2;
    char type = 't';
public:
    Apple(int _no, double _weight, char _key, int _size, char _type)
        : Fruit(_no, _weight, _key), size(_size), type(_type) {
        cout << "Calling Apple(int, double, char, int, char)..." << endl;
    }
    Apple() {
        cout << "Calling Apple()..." << endl;
    }
    void save() {   }
    virtual void process(){ cout << "Apple::process()" << endl; }
    virtual ~Apple() {
        cout << "Calling ~Apple()..." << endl;
    }
};

void* myAlloc(size_t size) {
    return malloc(size);
}
void myFree(void* ptr) {
    return free(ptr);
}

inline void* operator new(size_t size) {
    cout << "global new()... \n"; return myAlloc(size);
}
inline void operator delete(void* ptr) {
    cout << "global delete()... \n"; myFree(ptr);
}

int main() {
    cout << "---------------------------------------------------" << endl;
    cout << "size of int: " << sizeof(int) << ", "
         << "size of double: " << sizeof(double) << ", "
         << "size of char: " << sizeof(char) << endl;
    cout << "size of Fruit: " << sizeof(Fruit) << endl;
    cout << "size of Apple: " << sizeof(Apple) << endl;

    Fruit fr;
    Apple ap;
    cout << "Address of Vptr in fr: " << (int*)(&fr) << endl;
    cout << "Address of Vptr in ap: " << (int*)(&ap) << endl;

    cout << "Address of Vtbl in fr: " << *((int*)(&fr)) << endl;
    cout << "Address of Vtbl in ap: " << *((int*)(&ap)) << endl;

    cout << "Address of Vfunc of fr: " << (int*)(*((int*)(&fr))) << endl;
    cout << "Address of Vfunc of ap: " << (int*)(*((int*)(&ap))) << endl;


    typedef void(*Fun)(void);
    Fun pFun = NULL;
    pFun = (Fun)*((int*)(*((int*)(&fr))));
    pFun();
    pFun = (Fun)*((int*)(*((int*)(&ap))));
    pFun();

    cout << endl << endl;
    cout << "------Display Data Member of Fruit Object fr: " << endl;

    cout << "Address of no: " << (int*)(&fr) + 1 << endl;
    cout << "no = " <<  *((int*)(&fr) + 1) << endl;

    cout << "Address of weight: " << (double*)((int*)(&fr) + 2) << endl;
    cout << "weight = " << *((double*)((int*)(&fr) + 2)) << endl;

    cout << "Address of key: " << (int*)(((double*)((int*)(&fr) + 2)) + 1) << endl;
    cout << "key = " << (char)*(int*)(((double*)((int*)(&fr) + 2)) + 1)<< endl;

    cout << "------Display Data Member of Apple Object ap: " << endl;

    cout << "Address of no: " << (int*)(&ap) + 1 << endl;
    cout << "no = " <<  *((int*)(&ap) + 1) << endl;

    cout << "Address of weight: " << (double*)((int*)(&ap) + 2) << endl;
    cout << "weight = " << *((double*)((int*)(&ap) + 2)) << endl;

    cout << "Address of key: " << (int*)(((double*)((int*)(&ap) + 2)) + 1) << endl;
    cout << "key = " << (char)*(int*)(((double*)((int*)(&ap) + 2)) + 1)<< endl;

    cout << "Address of size: " << (int*)(((double*)((int*)(&ap) + 2)) + 1) + 1 << endl;
    cout << "size = " << *((int*)(((double*)((int*)(&ap) + 2)) + 1) + 1) << endl;

    cout << "Address of type: " << (int*)(((double*)((int*)(&ap) + 2)) + 1) + 2  << endl;
    cout << "type = " << (char)*((int*)(((double*)((int*)(&ap) + 2)) + 1) + 2) << endl;
    cout << "----------------------------------------------------" << endl;

    Fruit fru = Fruit(4, 3.5, 'l');
    Apple app = Apple(6, 4.7, 'm', 2, 'u');
    cout << "----------------------------------------------------" << endl;
    Apple* pApp = new Apple();
    delete pApp;
    cout << "----------------------------------------------------" << endl;

    return 0;
}


以GNU GCC Complier进行测试,运行结果如下图:

程序运行结果.png

对象内存分布图如下:

对象内存分布图.png

图中未画出RTTI相关信息。

由于以8个字节为存储单位进行内存对齐,在对象 fr 的成员 key 之后存在4个字节的空间,使得内存对齐,同样地,在对象 ap 的成员 type 之后也有4个字节的空间以使内存对齐。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,362评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,330评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,247评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,560评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,580评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,569评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,929评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,587评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,840评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,596评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,678评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,366评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,945评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,929评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,165评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,271评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,403评论 2 342

推荐阅读更多精彩内容