在面向对象的 C++ 语言中,虚函数(virtual function)是一个非常重要的概念。
什么是虚函数:
虚函数是指一个类中你希望重写的成员函数 ,当你用一个 父类(基类)指针或引用 指向一个继承类对象的时候,调用一个虚函数时, 实际调用的是子类(继承类)的版本。 ——摘自MSDN
#include <iostream>using namespace std;
class Parent
{
public:
char data[20];
void Function1();
//这里 virtual 声明 Function2是虚函数
virtual void Function2();
}parent;
void Parent::Function1()
{
printf("This is parent,function1\n");
}
void Parent::Function2()
{
printf("This is parent,function2\n");
}
class Child:public Parent
{
void Function1();
void Function2();
} child;
void Child::Function1()
{
printf("This is child,function1\n");
}
void Child::Function2()
{
printf("This is child,function2\n");
}
int main(int argc, char* argv[])
{
Parent *p; // 定义一个基类指针
if(_getch()=='c') // 如果输入一个小写字母c
p=&child; // 指向继承类对象
else
p=&parent; // 否则指向基类对象
p->Function1(); // 这里在编译时会直接给出Parent::Function1()的入口地址。
p->Function2(); // 注意这里,执行的是哪一个Function2?
return 0;
}
用任意版本的 Visual C++ 或 Borland C++ 编译并运行,输入一个小写字母c,得到下面的结果:
1 This is parent,function1
2 This is child,function2
第一行打印:因为我们是用一个Parent类的指针调用函数 Fuction1(),虽然实际上这个指针指向的是Child 类的对象,但编译器无法知道这一事实(直到运行的时候,程序才可以根据用户的输入判断出指针指向的对象),它只能按照调用 Parent 类的函数来理解并编译,所以我们看到了第一行的结果。
第二行打印:Function2() 函数在父类(基类)中被virtual关键字修饰,也就是说,它是一个虚函数。虚函数最关键的特点是“动态联编”,它可以在运行时判断指针指向的对象,并自动调用相应的函数。
如果我们在运行上面的程序时任意输入一个非c的字符,结果如下:
1 This is parent,function1
2 This is parent,function2
请注意看第二行,它的结果出现了变化。程序中仅仅调用了一个Function2()函数,却可以根据用户的输入自动决定到底调用基类中的Function2还是继承类中的Function2,这就是虚函数的作用。
Java中的虚函数
- Java 中普通函数就是虚函数(同等于C语言中virtual关键词修饰的方法)
- Java 中其实没有虚函数的概念,它的普通函数就相当于C++的虚函数,动态绑定是Java的默认行为。如果Java中不希望某个函数具有虚函数特性,可以加上final关键字变成非虚函数。C++中普通成员函数加上virtual关键字就成为虚函数,
- 虚函数的存在是为了多态
- 据统计Java的:静态方法、私有方法、final方法、实例构造器、父类方法都是非虚方法,除此之外都是虚方法
Java里面 final 关键字用法巩固
- fianl 修饰的类不可以被继承,即不允许被继承
- final 修饰成员变量之后就意味着该成员变量不能够被修改,由于外界无法修改该成员变量的值,故而其创建时就需要被赋值或者在其构造方法处赋值。如果final变量没有被赋值则会报错,且只能赋值一次
- final 修饰引用,引用被final修饰之后,虽然不能再指向其他对象,但是它指向的对象的内容是可变的,即虽然我们的这个应用不能再指向其它的对象了,但是该引用指向的这个对象的内容仍然是可以被修改的
- final 不能用来修饰构造方法
- final 用来修饰方法之后该方法无法被子类重写
Java中 父类 final 修饰的方法不能被子类重写,父类 static 修饰的方法不能被子类重写。