虚函数和多态
https://www.bilibili.com/video/BV1LK411s7ES
首先明确一个空类产生的对象的大小为1B:
class A {
};
int main() {
A a;
cout << sizeof(a) << endl;
}
然后我们向类A中加入两个普通成员函数,对象的大小还是1B:
class A {
public:
void func1() {}
void func2() {}
};
int main() {
A a;
cout << sizeof(a) << endl;
}
然而一旦我们向A中加入一个虚函数,其对象的大小变为8B:
class A {
public:
void func1() {}
void func2() {}
virtual void vfunc() {}
};
int main() {
A a;
cout << sizeof(a) << endl;
}
原因:当引入虚函数后,编译器在类中插入了一个虚函数表指针vptr,类似于下面的伪码:
class A{
public:
void* vptr;
...
};
而vtpr是占用类对象的内存空间的:
(gdb) p a
$1 = (A) {_vptr.A = 0x7ff771b64520 <vtable for A+16>}
每个类对象的虚函数表指针vptr 指向这个类的虚函数表vtbl ,编译器在编译期间在类A的构造函数内安插vptr 的赋值语句:
class A {
public:
A() {
vptr = &A::vtbl;
...
}
void* vptr;
};
考虑如下的A类对象的内存布局:
class A {
public:
void func1() {}
void func2() {}
virtual void vfunc() {}
virtual void vfunc2() {}
virtual ~A() {}
private:
int m_a;
int m_b;
};
(gdb) p a
$1 = (A) {_vptr.A = 0x7ff77a105520 <vtable for A+16>, m_a = 0, m_b = 1}
虚析构的作用
注意上面的例子中把A的析构函数设为虚函数,在实际开发中,这样做的目的是保证当用一个基类的指针删除一个派生类的对象时,派生类的析构函数会被调用,否则其只会调用基类的析构函数,如果派生类中有指针成员持有堆区内存,就得不到释放而造成内存泄漏
例如,这是正确的形式:
#include <iostream>
using namespace std;
class Base {
public:
virtual ~Base() {
cout << "Base dtor" << endl;
}
};
class Derived : public Base {
public:
~Derived() {
cout << "Derived dtor" << endl;
}
};
int main () {
Base* pb = new Derived();
delete pb;
return 0;
}
当delete父类指针时,子类的dtor也一同被调用:
daniel@ubuntu:~/project/demo$ g++ virtual-dtor.cpp
daniel@ubuntu:~/project/demo$ ./a.out
Derived dtor
Base dtor
如果去掉父类dtor前的virtual,则只会调用子类的dtor:
#include <iostream>
using namespace std;
class Base {
public:
~Base() {
cout << "Base dtor" << endl;
}
};
class Derived : public Base {
public:
~Derived() {
cout << "Derived dtor" << endl;
}
};
int main () {
Base* pb = new Derived();
delete pb;
return 0;
}
daniel@ubuntu:~/project/demo$ g++ virtual-dtor.cpp
daniel@ubuntu:~/project/demo$ ./a.out
Base dtor
多态:当通过父类指针指向子类对象,或通过父类引用绑定子类对象,调用父类中的虚函数,实际调用的是对应子类的虚函数
class Base {
public:
virtual void myvirfunc() {}
};
int main() {
Base* pb = new Base();
pb->vfunc();
Base b;
b.vfunc();
Base* pb2 = &b;
pb2->vfunc();
}
多态的表现:
class Base {
public:
virtual void myvirfunc() {}
};
class Derive : public Base {
public:
virtual void myvirfunc() {}
};
int main() {
Derive d;
Base* pb = &d;
pb->myvirfunc();
Base* pb2 = new Derive();
pb2->myvirfunc();
Derive d2;
Base& rb = d2;
rb.myvirfunc();
}
存在继承关系时虚函数表指针指向:
设父类有f(), g(), h() 这三个虚函数,子类重写了父类中的g() 虚函数,则父子类对象的虚函数表指针和虚函数表如下:
延申思考题:
- 当进行多重继承时,子类对象有几个虚函数表指针?子类对象有几个虚函数表?
- 虚基类表指针在对象内存中的布局?
用C语言模拟多态
https://www.bilibili.com/video/BV15g4y1a7F3
对于下面的多态实例:
#include <iostream>
using namespace std;
class ISpeaker {
protected:
int b;
public:
ISpeaker (int bb) : b(bb) {}
virtual void speak() = 0;
};
class Dog : public ISpeaker {
public:
Dog() : ISpeaker(0) {}
virtual void speak() override {
printf("woof %d\n", b);
}
};
class Human : public ISpeaker {
private:
int c;
public:
Human() : ISpeaker(1), c(2) {}
virtual void speak() override {
printf("hello %d\n", c);
}
};
int main(){
ISpeaker* d = new Dog();
ISpeaker* h = new Human();
d->speak();
h->speak();
return 0;
}
其用C语言可以描述如下:
#include <iostream>
using namespace std;
extern "C" {
struct vft {
void (*speak) (void* ptr);
};
void Dog_speak (void* ptr) {
void* p = ptr + sizeof(vft*);
int bb = *((int*)p);
printf("woof %d\n", bb);
}
void Human_speak (void* ptr) {
void* p = ptr + sizeof(vft) + sizeof(int);
int cc = *((int*)p);
printf("hello %d\n", cc);
}
const static vft Dog_vft= {
.speak = Dog_speak
};
const static vft Human_vft = {
.speak = Human_speak
};
struct ISpeaker {
const vft* vptr;
int b;
};
struct Dog {
const vft* vptr;
int b;
};
struct Human {
const vft* vptr;
int b;
int c;
};
Dog* Dog_constructor() {
Dog* d =(Dog*)malloc(sizeof(Dog));
d->vptr = &Dog_vft;
d->b = 0;
return d;
}
Human* Human_constructor() {
Human* h =(Human*)malloc(sizeof(Human));
h->vptr = &Human_vft;
h->b = 1;
h->c = 2;
return h;
}
int main () {
ISpeaker* s1 = (ISpeaker*)Dog_constructor();
ISpeaker* s2 = (ISpeaker*)Human_constructor();
s1->vptr->speak(s1);
s2->vptr->speak(s2);
return 0;
}
}
|