覆写
一个实例方法可以覆写(override)在其超类中可访问到得具有相同签名的所有实例方法
[JLS 8.4.8.1],从而使能了动态分派(dynamic dispatch);换句话说,VM将给予实例的
运行期类型来悬着调用的覆写方法[JLS 15.12.4.4]。覆写是面向对象编程技术的基础,
并且是惟一没有被普遍劝阻的名字重用形式:
class Base { public void f(){} } class Derived extends Base{ public void f(){} // overrides Base.f() }
隐藏
一个域、静态方法或成员类型可以分别隐藏(hide)在其超类中可访问到得具有相同名字
(对方法而言就是相同的方法签名)的所有域、静态方法或成员类型。
隐藏成员将阻止其被继承[JLS 8.3,8.2.8.2,8.5]
class Base { public static void f(){} } class Derived extends Base{ public static void f(){} // hides Base.f() }
重载
在某个类中的方法可以重载(overload)另一个方法,只要它们具有相同的名字和不同的签名。
由调用所指定的重载方法是在编译期选定的[JSL 8.4.9, 15.12.2]
class CircuitBreaker { public void f(int i) {} // int overloading public void f(String s){} // String overloading }
遮蔽
一个变量、方法或类型可以分别遮蔽(shadow)在一个闭合的文本范围内的
具有相同名字的所有变量、方法或类型。如果一个实体被遮蔽了,那么你用它
的简单名是无法应用到它的;根据实体的不同,有时你根本就无法引用到它。
class WhoKonws { static String sentence = "I don't know"; public static void main(String[] args) { String sentence = "I don't know"; //shadows static field System.out.println(sentence); //prints local variable } }
尽管遮蔽通常是被劝阻的,但是有一种通用的惯用法确实涉及遮蔽。
--》构造器,普遍认为这种风格实惠超过风险。
class Belt { private final int size; public Belt(int size){ // Parameter shadows Belt.size this.size = size; } }
遮掩
一个变量可以遮掩(obscure)具有相同的名字的一个类型,只要它们都在同一个范围内:
如果这个名字被用于变量与类型都被许可的范围,那么它将引用到变量上。相似的,
一个变量或一个类型可以遮掩一个包。遮掩是惟一一种两个名字位于不同的名字空间的
名字重用形式,这些名字空间包括:变量、包、方法或类型。如果一个类型或一个包
被遮掩了,那么你不能通过其简单名引用到它,除非是这样一个上下午环境中,
即语法只允许在其名字空间中出现一种名字。遵守命名习惯可以极大地消除产生遮掩的可能性。
class Obscure { private String System; public static void main(String[] args) { //Next line won't compile :System refers to static field System.out.println("hello,obscure world"); } }