java的覆盖和隐藏的区别
2014-03-17 11:18
183 查看
我们知道,在JAVA中,子类可以继承父类,如果子类声明的方法与父类有重名的情况怎么办,大伙儿都知道要是重写,但是实际上这又分为两种情况,就是方法和变量在继承时的覆盖和隐藏问题,这些概念性的东西看似无聊,但是在面试或者是SCJP认证题中围绕这些是会经常碰到的,所以这里来讨论下
首先我们来看几个概念
隐藏:child隐藏了parent的变量和方法,那么,child不能访问parent被隐藏的变量或者方法,但是,讲B转换成A中,可以访问A被隐藏的变量或者方法
覆盖:child覆盖了parent的变量或者方法,那么,child不能访问parent被覆盖的变量或者方法,将child转换成parent后同样不能访问parent被覆盖的变量或者方法
覆盖:子类重写父类的方法,要求方法名和参数类型完全一样(参数不能是子类),返回值和异常比父类小或者相同(即为父类的子类),访问修饰符比父类大或者相同
两同两小一大
注意:子类实例方法不能覆盖父类的静态方法;子类的静态方法也不能覆盖父类的实例方法(编译时报错),总结为方法不能交叉覆盖
隐藏:父类和子类拥有相同名字的属性或者方法时,父类的同名的属性或者方法形式上不见了,实际是还是存在的
注意:当发生隐藏的时候,声明类型是什么类,就调用对应类的属性或者方法,而不会发生动态绑定
方法隐藏只有一种形式,就是父类和子类存在相同的静态方法
属性只能被隐藏,不能被覆盖
子类实例变量/静态变量可以隐藏父类的实例/静态变量,总结为变量可以交叉隐藏
隐藏和覆盖的区别:
被隐藏的属性,在子类被强制转换成父类后,访问的是父类中的属性
被覆盖的方法,在子类被强制转换成父类后,调用的还是子类自身的方法
特殊情况:
1.final修饰的属性可以被隐藏,但是不能被赋值,即不能用=来赋值,网上说final属性不能被修改,这个说法不准确,因为对于引用类型的变量用final修饰后,它只是不能被指向其他对象,但是可以改它自身的值,可以用ArrayList测试,final属性可以在运行的时候进行初始化,但是不能不出现初始化语句
2.final修饰的方法不能被覆盖,可以被重载
3.final修饰的类不能被继承
4.private方法隐式添加了final
首先我们来看几个概念
隐藏:child隐藏了parent的变量和方法,那么,child不能访问parent被隐藏的变量或者方法,但是,讲B转换成A中,可以访问A被隐藏的变量或者方法
覆盖:child覆盖了parent的变量或者方法,那么,child不能访问parent被覆盖的变量或者方法,将child转换成parent后同样不能访问parent被覆盖的变量或者方法
覆盖:子类重写父类的方法,要求方法名和参数类型完全一样(参数不能是子类),返回值和异常比父类小或者相同(即为父类的子类),访问修饰符比父类大或者相同
两同两小一大
注意:子类实例方法不能覆盖父类的静态方法;子类的静态方法也不能覆盖父类的实例方法(编译时报错),总结为方法不能交叉覆盖
隐藏:父类和子类拥有相同名字的属性或者方法时,父类的同名的属性或者方法形式上不见了,实际是还是存在的
注意:当发生隐藏的时候,声明类型是什么类,就调用对应类的属性或者方法,而不会发生动态绑定
方法隐藏只有一种形式,就是父类和子类存在相同的静态方法
属性只能被隐藏,不能被覆盖
子类实例变量/静态变量可以隐藏父类的实例/静态变量,总结为变量可以交叉隐藏
隐藏和覆盖的区别:
被隐藏的属性,在子类被强制转换成父类后,访问的是父类中的属性
被覆盖的方法,在子类被强制转换成父类后,调用的还是子类自身的方法
特殊情况:
1.final修饰的属性可以被隐藏,但是不能被赋值,即不能用=来赋值,网上说final属性不能被修改,这个说法不准确,因为对于引用类型的变量用final修饰后,它只是不能被指向其他对象,但是可以改它自身的值,可以用ArrayList测试,final属性可以在运行的时候进行初始化,但是不能不出现初始化语句
2.final修饰的方法不能被覆盖,可以被重载
3.final修饰的类不能被继承
4.private方法隐式添加了final
classBase{
intx=1;
staticinty=2;
intz=3;
intmethod(){
returnx;
}
}
classSubclassextendsBase{
intx=4;
inty=5;
staticintz=6;
intmethod(){
returnx;
}
}
publicclassTest{
publicstaticvoidmain(String[]args){
Subclasss=newSubclass();
System.out.println(s.x+""+s.y+""+s.z);
System.out.println(s.method());
Baseb=(Subclass)s;
System.out.println(b.x+""+b.y+""+b.z);
System.out.println(b.method());
}
}
结果:456
4
123
4
分析:在main方法中,开始创建了一个subclass对象实例,对于第一个输出很好理解;后将对象实例s强制转换成父类Base类型,由于父类的成员变量只是被隐藏,所以通过强制转换成父类类型的对象实例调用时,调用的是父类中的变量,而method方法是被子类同名方法所覆盖,所以调用时依然调用的子类method方法。
相关文章推荐
- Java多线程(一)、多线程的基本概念和使用
- java 静态方法不能被override
- Myeclipse/Eclipse中修改WebContent/WebRoot
- java类初始化顺序
- JAVA_银行业务调度系统
- spring aop 实现读写分离
- javaweb之jstl+el完成数据迭代
- Java进阶篇--容器(二)
- JAVA_交通灯管理系统
- java 中 class类作用
- JAVA虚拟机体系结构
- StringUtils
- Spring整合jaxws基于handles验证
- 使用myeclipse时出现java.lang.UnsupportedClassVersionError: Bad version number in .class file
- java字符串截取
- struts2标签#、%、$取值
- java服务端远程监控辅助参数
- java发邮件的代码
- Struts2之数据校验validate()函数
- activeMQ与spring结合 使用及配置