我参加了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()
方法,或者任何其他子类唯一的方法。
答案 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的方法将覆盖超类中的方法。 此技巧的缺点是,您必须为要从子类调用的每个方法在超类中声明一个空方法,如果要添加多个方法,则该方法可能会变得混乱。因此,仅对将在许多不同子类中使用的方法执行此操作。