第51课 C++对象模型分析(下)
2018-01-22 10:29
363 查看
1、单继承对象模型
1.1、单一继承
在C++编译器的内部类可以理解为结构体
子类是由父类成员叠加子类新成员得到的。
class Derived : public Demo
{
int mk;
};
/************************* 继承对象模型初探 ********************/
#include <iostream>
using namespace std;
class Demo
{
protected:
int mi;
int mj;
public:
//虚函数
virtual void print()
{
cout << "mi = " << mi << ", "
<< "mj = " << mj << endl;
}
};
class Derived : public Demo
{
int mk;
public:
Derived(int i, int j, int k)
{
//说明那个虚函数的指针在这里,要对齐。
mi = i;
mj = j;
mk = k;
}
void print()
{
cout << "mi = " << mi << ", "
<< "mj = " << mj << ", "
<< "mk = " << mk << endl;
}
};
struct Test
{
void* p; //说明那个虚函数的指针在这里,要对齐。
int mi;
int mj;
int mk;
};
int main()
{
cout << "sizeof(Demo)" << sizeof(Demo) << endl;//12, 不是8,因为插入了一个虚函数表指针(指针长度4字节)
cout << "sizeof(Derived)" << sizeof(Derived) << endl;//16,不是12,因为继承了父类,原因同上。
Derived d(1, 2, 3);
Test* p = reinterpret_cast<Test*>(&d);//指针之间的转换,
cout << endl;
//以下实验证明带有虚函数的Derived的内存模型与Test结构体是一致的!!!!!
//1,大小相同,2。第1个成员变量vptr(虚函数)指针;3。往后依次为mi,mj,mk
cout << "Before Change..." << endl;
d.print();
p->mi = 10;
p->mj = 20; //通过内存地址直接访问类里面的变量。
p->mk = 30;
cout << "After Change..." << endl;
d.print();
return 0;
}
2、C++多态的实现原理
2.1、当类中声明虚函数时,编译器会在类中生成一个虚函数表2.2、虚函数表是一个存储成员函数地址的数据结构
2.3、虚函数表是由编译器自动生成与维护的
2.4、virtual成员函数会被编译器放入虚函数表中
2.5、存在虚函数时,每个对象中都有一个指向虚函数表的指针
多态的本质分析::用C写面向对象,实现多态
#ifndef _51_2_H_
#define _51_2_H_
typedef
void
Demo;
typedef
void
Derived;
//父类
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);
//子类
Derived*
Derived_Create(int
i,
int
j,
int
k);
//子类的构造函数。
int
Derived_GetK(Derived*
pThis);
int
Derived_Add(Derived*
pThis,
int
value);
#endif
#include "51-2.h"
#include <malloc.h>
#include <stdio.h>
//利用C语言来实现面向对象..实现多态的效果。
//static 修饰函数,说明函数只能被本文件所使用。和修饰全局变量一样,限定在本文件作用域,隐藏的虚函数表。
static int Demo_Virtual_Add(Demo* pThis, int value);
static int Derived_Virtual_Add(Derived* pThis, int value);
struct VTable //2.定义虚函数表数据结构
{//这是一张表
int (*pAdd)(void*, int); //3.虚函数表里面存储的内容是(函数指针),,主要这个pAdd具体指向谁,是地址层面的具体数值。
};
struct ClassDemo //父类结构体
{
struct VTable* vptr; //1.数定义虚函表指针 ==》 虚函数表指针的类型???
int mi;
int mj;
};
struct ClassDerived//子类结构体,拥有父类结构体里面的成员。也包括那个虚函数表指针
{
struct ClassDemo d; //模拟继承,实质就是父类成员叠加到子类成员中去。
int mk;
};
//父类的虚函数表,static隐藏到当前文件中。//给虚函数表初始化
static struct VTable g_Demo_vtbl = //g_Demo_vtbl的类型为static struct VTable
{
Demo_Virtual_Add //内部都是一个函数指针,函数名。
};
//子类的虚函数表
static struct VTable g_Derived_vtbl = //静态结构体变量。
{
Derived_Virtual_Add//内部都是一个函数指针
};
/****************** 父类 ***********************/
Demo*
f4c7
Demo_Create(int i, int j) //相当于构造函数
{
struct ClassDemo* ret = (struct ClassDemo*)malloc(sizeof(struct ClassDemo));
if(ret != 0)
{
//4、、只要有父类的对象产生,那么调用构造函数,关联虚函数表。
ret->vptr = &g_Demo_vtbl;//4、、(g_Demo_vtbl)结构体类型,关联对象和虚函数表
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;
}
//6.定义虚函数表中指针所指向的具体函数
static int Demo_Virtual_Add(Demo* pThis, int value)
{
struct ClassDemo* obj = (struct ClassDemo*)pThis;
return obj->mi + obj->mj + value;
}
//5.分析具体的虚函数
int Demo_Add(Demo* pThis, int value) //是个虚函数
{
struct ClassDemo* obj = (struct ClassDemo*)pThis;
//从虚函数表中找到真正的实现函数
return obj->vptr->pAdd(pThis, value); //虚函数表具体的指向,这个vptr在对象的构造函数中初始化。
}
void Demo_Free(Demo* pThis)
{
free(pThis);
}
/****************** Derived类 ,子类 ***********************/
Derived* Derived_Create(int i, int j, int k) //构造函数
{
struct ClassDerived* ret = (struct ClassDerived*)malloc(sizeof(struct ClassDerived));
if(ret != NULL)
{
//只要有子类的对象产生,那么调用构造函数,关联虚函数
//(g_Derived_vtbl)结构体类型,关联对象和虚函数表
ret -> d.vptr = &g_Derived_vtbl; //结构体指针,可以访问结构体成员。
ret -> d.mi = i;
ret -> d.mj = j;
ret -> mk = k;
}
return ret;
}
int Derived_GetK(Derived* pThis)
{
struct ClassDerived* obj = (struct ClassDerived*)pThis;
return obj->mk;
}
//定义子类虚函数表中指针所指向的具体函数
static int Derived_Virtual_Add(Derived* pThis, int value)
{
struct ClassDerived* obj = (struct ClassDerived*)pThis;
return obj->mk + value;
}
//分析虚函数
int Derived_Add(Derived* pThis, int value)
{
struct ClassDerived* obj = (struct ClassDerived*)pThis;
//从虚函数表中找到真正的实现函数
return obj->d.vptr->pAdd(pThis, value);
}
#include <stdio.h>
#include "51-2.h"
void run(Demo* p, int v)
{
int r = Demo_Add(p, v); //多态
printf("r = %d\n", r);
}
int main()
{ //这两个指针类型都是void的。
Demo* pb = Demo_Create(1, 2);
Derived* pd = Derived_Create(1, 22, 333);
printf("pb->add(3) = %d\n", Demo_Add(pb, 3));
printf("pd->add(3) = %d\n", Derived_Add(pd, 3));
run(pb, 3);
run(pd, 3);
Demo_Free(pb);
Demo_Free(pd);
return 0;
}
C语言实现多态的思路
5、小结
5.1、继承的本质是父子间成员变量的叠加5.2、C++中的多态是通过虚函数表实现的
5.3、虚函数表是由编译器自动生成与维护的
5.4、虚函数的调用效率低于普通成员函数。(虚函数要经过几次寻址)
【参考资料】:C++对象模型之详述C++对象的内存布局
课程中通过
void* 指针保证具体的结构体成员是不能在外界被访问的, 以
此模拟 C++ 中 private
和 protected。 因此,在头文件中定义了如下的语句:
typedef void
Demo;
typedef void
Derived;
Demo 和 Derived 的本质依旧是
void, 所以,用 Demo* 指针和 Derived* 指针
指向具体的对象时,无法访问对象中的成员变量, 这样就达到了“外界无法访问类
中私有成员” 的封装效果!
继承的本质是父类成员与子类成员的叠加,所以在用 C 语言写面向对象程
序的时候, 可以直接考虑结构体成员的叠加即可。课程中的实现直接将
struct
ClassDemo d 作为
struct ClassDerived 的第一个成员,
以此表现两个自定义数
据类型间的继承关系。 因为
struct
ClassDerived 变量的实际内存分布就是由
struct
ClassDemo 的成员以及
struct
ClassDerived 中新定义的成员组成的, 这
样就直接实现了继承的本质, 所以说
struct ClassDerived 继承自
struct
ClassDemo。
下一步要实现的就是多态了!
多态在 C++ 中的实现本质是通过虚函数表完
成的,
而虚函数表是编译器自主产生和维护的数据结构。因此,接下来要解决的问
题就是如何在 C 语言中自定义虚函数表?课程中认为通过结构体变量模拟 C++
中的虚函数表是比较理想的一种选择,所以有了下面的代码:
struct
VTable
{
int
(*pAdd)(void*,
int);
};
必须要注意的是,课程中由于复制粘贴的缘故,误将 pAdd 指针的类型定义成了
int
(*)(Derived*,
int) , 这从 C 语言的角度算不上错误,因为 Derived* 的本质就
是
void* , 所以编译运行都没有问题。但是,从面向对象的角度,这里可以说是
一种语义上的错误! 因为 pAdd 必须可以指向父类中定义的 Add 函数版本,也可
以指向子类中定义的 Add 函数版本,所以说用 Derived* 作为第一个参数表示实
际对象并不合适,应该直接使用
void* 。
有了类型后就可以定义实际的虚函数表了,
在 C 语言中用具有文件作用域
的全局变量表示实际的虚函数表是最合适的,因此有了下面的代码:
//
父类对象使用的虚函数表
static struct
VTable g_Demo_vtbl =
{
Demo_Virtual_Add
};
//
子类对象使用的虚函数表
static struct
VTable g_Derived_vtbl =
{
Derived_Virtual_Add
};
每个对象中都拥有一个指向虚函数表的指针,而所有父类对象都指向
g_Demo_vtbl,所以所有子类对象都指向 g_Derived_vtbl。当一切就绪后,实际
调用虚函数的过程就是通过虚函数表中的对应指针来完成的。
相关文章推荐
- 第51课-C++对象模型分析(下)
- 29、不一样的C++系列--对象模型分析
- C++ - 多重继承和虚拟继承对象模型、效率分析
- c++对象内存模型分析工具
- C++ 多重继承和虚拟继承对象模型、效率分析
- 第50课 C++对象模型分析(上)
- C++多重继承和虚拟继承对象模型、效率分析
- C++多重继承和虚拟继承对象模型、效率分析
- C++对象模型和虚函数表分析以及重载、重写、隐藏的区别
- 第50课 - C++ 对象模型分析(上)
- C++多重继承和虚拟继承对象模型、效率分析
- 第50课-C++对象模型分析(上)
- 【C++】菱形虚拟继承对象模型分析
- C++对象模型分析
- Microsoft Visual C++虚拟多继承 对象模型初步分析
- C++对象模型中的虚拟函数分析
- c++对象模型布局分析
- C++对象模型Data语意学分析、虚继承底层实现机制
- 《深入探索C++对象模型》第二章:构造函数语意学(下)
- C++对象模型