一道关于继承的题目

问:以下代码段最后输出的是什么

class Dog {
	public static void bark() {
		System.out.println("woof ");
	}
}

class Basenji extends Dog {

	public static void bark() {
	}
}

public class Bark {
	public static void main(String args[]) {
		Dog woofer = new Dog();
		Dog nipper = new Basenji();
		woofer.bark(); // 
		nipper.bark(); // 
	}
}

答案是:

bark

bark

我们都知道:重写指的是根据运行时对象的类型来决定调用哪个方法,而不是根据编译时的类型。在这道题中,父类的引用指向子类的实例( Dog nipper = new Basenji(); ),运行的时候,本来调用的应该是子类的方法bark(),但是这道题却没有。这是为什么呢?

关键点在于父类的这个方法是静态的,静态方法可以说是类的方法,它们在编译阶段就使用编译出来的类型进行绑定了。使用对象引用来访问静态方法只是Java设计者给程序员的自由。我们应该直接使用类名来访问静态方法,而不要使用对象引用来访问(出自: http://www.importnew.com/7784.html)  静态方法即使使用实例来引用,依然是在底层是通过类名引用的 因此重写是无效的。所以这道题最后输出的是两个bark而不是一个。

重写无效,但是子类中的bark方法依然是存在的,这个事实不能否认。我们试一下用子类的对象的引用来调用这个方法就知道了。

class Dog {
	public static void bark() {
		System.out.println("woof ");
	}
}

class Basenji extends Dog {

	public static void bark() {
		System.out.println("hahaha");
	}
}

public class Bark {
	public static void main(String args[]) {
		Dog woofer = new Dog();
		//Dog nipper = new Basenji();
		Basenji nipper = new Basenji();
		woofer.bark(); // woof
		nipper.bark(); // hahaha
	}
}





  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值