*
在C++中:
重载的条件:
- 同一类中;
- 函数名称相同;
- 参数不同,包括个数和类型;
- virtual可有可无。
覆盖的条件:
- 派生类—-> 基类;
- 函数名相同;
- 参数相同,包括个数和类型;
- 基类函数有virtual修饰。
隐藏:派生类屏蔽了与其同名的基类函数
- 如果派生类的函数名和基类相同,但是参数不同,此时无论有无virtual修饰,基类的函数被隐藏;
- 如果派生类与基类函数名相同,而且参数也完全相同,但是基类的函数没有virtual修饰,则基类的函数被隐藏。
JAVA中:
- 同名的实例方法被覆盖(参数也完全一样)(多态的实现),同名的静态方法被隐藏;
- 覆盖和隐藏的区别在于,子类对象转换为父类对象后可以访问父类的被隐藏的方法,而不能访问被覆盖的方法;
- 在java中,在还未初始化子类的时候,子类的同名函数就已经覆盖父类的了;(有相应的博客说过这个问题)
- 成员属性只能被隐藏,不能被覆盖。
先上代码:
C++代码:
class A{
public:
virtual void foo1()
{
cout<<"在A中foo1()..."<<endl;
}
void foo2(int a)
{
cout<<"在A中foo2(int a) "<<a<<endl;
}
virtual void foo3(int a)
{
cout<<"在A中foo3(int a) "<<a<<endl;
}
void foo4(int a)
{
cout<<"在A中foo4(int a) "<<a<<endl;
}
};
class B :public A{
public :
void foo1()
{
cout<<"在B中foo1()..."<<endl;
}
void foo2(char a)
{
cout<<"在B中foo2(char a) "<<a<<endl;
}
void foo3(char a)
{
cout<<"在B中foo3(char a) "<<a<<endl;
}
void foo4(int a)
{
cout<<"在B中foo4(int a) "<<a<<endl;
}
void foo5()
{
cout<<"foo5..."<<endl;
}
};
int _tmain(int argc, _TCHAR* argv[])
{
A *ptr = new B();
ptr->foo1(); //多态,输出子类函数
ptr->foo2(100); //
ptr->foo2('a');
ptr->foo3(100);
ptr->foo3('b');
ptr->foo4(100);
((B*)ptr)->foo5();
}
分析:
- 基类的foo1()函数是被正确的重载了,所以输出的是派生类的实现;
- 按定义foo2()被隐藏,ptr是指向A的指针,所以输出的是基类的实现;
- 同样,foo3()被隐藏,ptr是指向A的指针,所以输出的是基类的实现;
- foo4()同上;
- 如果派生类新增了成员函数,则必须将基类的指针ptr强制转换成派生类对象才能调用;
如果把main函数中的A *ptr = new B(); 改成 B *ptr = new B();结果如下:
分析:
- foo1()多态的体现;
- 基类的foo2(),foo3(),foo4()均被隐藏,派生类的指针ptr访问不到;
java代码:
class A{
public void foo1()
{
System.out.println("在A中foo1()");
}
public void foo2(int a)
{
System.out.println("在A中foo2(int a) "+a);
}
public void foo3(int a)
{
System.out.println("在A中foo3(int a) "+a);
}
public static void foo4(int a)
{
System.out.println("在A中foo4(int a) "+a);
}
};
class B extends A{
public void foo1()
{
System.out.println("在B中foo1()...");
}
public void foo2(char a)
{
System.out.println("在B中foo2(char a) "+a);
}
public void foo3(char a)
{
System.out.println("在B中foo3(char a) "+a);
}
public static void foo4(int a)
{
System.out.println("在B中foo4(int a) "+a);
}
};
public class wang {
public static void main(String[] args) {
// TODO Auto-generated method stub
A ptr = new B();
ptr.foo1(); //多态,输出子类函数
ptr.foo2(100); //
ptr.foo2('a');
ptr.foo3(100);
ptr.foo3('b');
ptr.foo4(100);
A a=ptr;
a.foo4(10);
}
}
- foo1()是多态的实现,访问的是子类的实现;
- 父类的foo2(),foo3()均没有被子类覆盖,ptr是父类对象,所以访问的是父类的实现;
- foo4()属于同名的静态方法被隐藏,ptr是父类对象,所以访问的是父类的实现;
如果把主方法中的A ptr = new B(); 改成 B ptr = new B();结果为:
- 发现相同的代码在C++中和在java中的结果完全不一样;
- 父类的foo2()和foo3()没有被覆盖,当然也不是被隐藏,所以可以理解为它们在子类中和子类中对应的方法一同以重载出现,根据参数的不同,调用对应的方法;
关于面向对象的编程,主要三个方面就是封装、继承、多态。语言教程看了很多遍,但不久后便忘记。原因就是应用的少,没有在实战中提升对它们的深刻理解。
肯定有一些不准确的地方,待以后慢慢改正。