我不太明白为什么静态方法可以在Java中继承?
继承就像从基类继承AND静态属于Class而不是Object.
因此,如果静态属于类只有为什么它会逐渐渗透到派生类?
它不应该只停留在定义它的类中吗?
继承静态方法是一种很好的编程习惯吗?
解决方法:
在java中,静态方法不会被继承(或者正确的单词被覆盖),但它们可以被隐藏.
这里的不同之处在于它们不像对象方法那样受多态性影响.
public class C1 {
static public void M1() {
System.out.println("C1.M1().");
}
static public void main(String ... Args) {
M1();
}
}
public class C2 extends C1 {
static public void M1() {
System.out.println("C2.M1().");
}
static public void main(String ... Args) {
M1();
C1.main(Args);
}
}
运行C2.main(null)时,您将获得:
C2.M1().
C1.M1().
如你看到的,
在C1.main(…)中调用M1()是指C1和的M1
在C2.main(…)中调用M1()是指C2的M1.
调用M1(没有任何前缀,参见每个main()的第一行),不受多态性的影响,因为C1中的M1不会被C2覆盖.
但是从C2调用指的是C2的M1,因为C2的M1隐藏了C1中的M1.
阅读更多here.
编辑:我刚刚重新阅读你的问题,只看到关于“良好编程实践”的部分.
正如我所说,静态方法不是继承的,而是隐藏的,因此它们与不同的方法一样好.
从代码的角度来看,它们是完全不同的方法.
让我们说吧.
C1 has static method M1.
C2 extends C1 and has static method M1.
C3 extends C2.
从C1调用M1(无前缀)时,调用C1.M1().
从C2调用M1(无前缀)时,调用C2.M1(). //派生但隐藏起来
从C3调用M1(无前缀)时,调用C3.M1(). //派生而不是隐藏
要指定哪个方法,请使用类名称,如C1.M1(),C2.M1()和C3.M1()(这将称为C2.M1()).
因此,此实现允许重新实现静态方法,但仅作为不同的方法而不是重写(或替换)方法.
因此,这通常没有什么不同,比如命名不同,如:C1_M()和C2_M().
所以你可能会问,为什么还要这个功能呢?我真的不知道.也许允许更灵活的命名方法.
但我使用的用法(可能会或可能不会打算)是通过反射的多态性.
由于您可以使用反射按名称获取和调用方法,因此允许它们具有相同的名称将通过反射启用多态.
例如(rought代码,可能无法运行):
String aClsName = "C1"; // "C2";
Class aCls = Class.forName(aClsName);
Method aMth = aCls.getMethod("M1"); // M1 of C1 or C2 depends on the class name.
aMth.invoke(null);
要么
Object aObj = new C1(); // new C2();
Class aCls = aObj.getClass();
Method aMth = aCls.getMethod("M1"); // M1 of C1 or C2 depends on the class of the object.
aMth.invoke(null);
考虑一下,我认为Java也使用它(比如,writeObject(…)进行序列化)所以它可能是有意的.
总而言之,隐藏静态方法并不是一个好的编程实践(Eclipse也建议不要这样做),但它在两种情况下很有用,(1)命名方法完全符合它的预期,(2)使用它来变形它反射.
希望这可以帮助.