从超类调用子类方法

时间:2012-04-05 01:43:30

标签: java inheritance

我参加了java课程,我们刚刚开始学习继承。我正在开展一项任务,要求我们创建一个名称和年龄的“Pet”超类;和三个子类,每个子类都有自己独特的特性(我选择了“狗”,“猫”和“鸟”)。在我们构建了所有这些之后,我们将创建一个Main类来测试所有内容,这就是我遇到问题的地方。我试图在get中为这些独特的特征调用Main方法,但它似乎只能找到超类中的方法。

这是Main类:

public class Kennel {
    public static void main(String[] args) {
        // Create the pet objects
        Pet cat = new Cat("Feline", 12, "Orange");
        Pet dog = new Dog("Spot", 14, "Dalmation");
        Pet bird = new Bird("Feathers", 56, 12);

        // Print out the status of the animals
        System.out.println("I have a cat named " + cat.getName()
                + ". He is " + cat.getAge() + " years old."
                + " He is " + cat.getColor()
                + "When he speaks he says " + cat.speak());
        System.out.println("I also have a dog named " + dog.getName()
                + ". He is " + dog.getAge() + " years old."
                + " He is a " + dog.getBreed()
                + " When he speaks he says " + dog.speak());
        System.out.println("And Finally I have a bird named " 
                + bird.getName() + ". He is " + bird.getAge() + " years old."
                + " He has a wingspan of " + bird.getWingspan() + " inches."
                + " When he speaks he says " + bird.speak());       
    }
}

这是我的超类

abstract public class Pet {
    private String name;
    private int age;

    // Constructor
    public Pet(String petName, int petAge) {
        this.name = petName;
        this.age = petAge;
    }

    // Getters
    public String getName() { return(this.name); }
    public int getAge() { return(this.age); }

    // Setters
    public void setName(String nameSet) { this.name = nameSet; }
    public void setAge(int ageSet) { this.age = ageSet; }

    // Other Methods
    abstract public String speak();

    // toString
    @Override
    public String toString() {
        String answer = "Name: " + this.name + " Age: " + this.age;
        return answer;
    }
}

这里是其中一个子类(它们看起来都一样,并且有相同的错误)

public class Cat extends Pet {
    private String color;

    // Constructor
    public Cat(String petName, int petAge, String petColor) {
        super(petName, petAge);
        this.color = petColor;
    }

    // Getters
    public String getColor() { return(this.color); }

    // Setters
    public void setColor(String colorSet) { this.color = colorSet; }

    // Other Methods
    @Override
    public String speak() { return "Meow!"; } 

    // toString
    @Override
    public String toString() {
        String answer = "Name: " + super.getName() + " Age: "+super.getAge()
                + " Color: " + this.color;
        return answer;
    }
}

所以发生的事情是我无法获得主要方法来查找cat.getColor()方法,或者任何其他子类唯一的方法。

5 个答案:

答案 0 :(得分:24)

当您将变量声明为具有超类的类型时,您只能通过该变量访问超类​​的(公共)方法和成员变量。

Pet cat = new Cat("Feline",12,"Orange"); 
cat.getName(); // this is OK
cat.getColor(); // this is not OK, getColor() is not in Pet

要访问具体类中的方法(在本例中为Cat),您需要将变量声明为派生类

Cat cat = new Cat("Feline",12,"Orange"); 
cat.getName(); // OK, getName() is part of Cat (and the superclass)
cat.getColor(); // OK, getColor() is part of Cat

或者将它投射到你知道的类型/怀疑是具体类型

Pet cat = new Cat("Feline",12,"Orange"); 
((Cat)cat).getName(); // OK (same as above)
((Cat)cat).getColor(); // now we are looking at cat through the glass of Cat

您甚至可以将这两种方法结合起来:

Pet pet = new Cat("Feline",12,"Orange"); 
Cat cat = (Cat)pet;
cat.getName(); // OK
cat.getColor(); // OK

答案 1 :(得分:1)

Pet cat = new Cat("Feline",12,"Orange");
^^^
This is the error.

Pet没有名为getColor()

的方法

你需要这样做:

Cat cat = new Cat(...);

答案 2 :(得分:1)

执行此操作时:

 Pet cat = new Cat("Feline",12,"Orange");

编译器将变量cat视为Pet。所以你不能使用Cat classe的具体方法。

您必须将cat声明为Type Cat以解决您的问题。

问候。

答案 3 :(得分:1)

在这里,您尝试使用构造函数(Cat)创建超类类型对象(cat),这是一种子类方法,这是不可能的。这违反了继承规则。

Pet cat = new Cat("Feline", 12, "Orange");

答案 4 :(得分:0)

您还可以将这些空方法添加到Pet类中:

public String getColor() { }
public String setColor() { }
public String speak() { }

这样,您可以从Pet对象调用此方法,而不必将其强制转换为Cat。一旦被调用,Cat的方法将覆盖超类中的方法。 此技巧的缺点是,您必须为要从子类调用的每个方法在超类中声明一个空方法,如果要添加多个方法,则该方法可能会变得混乱。因此,仅对将在许多不同子类中使用的方法执行此操作。