【问题标题】:Can only access Parent class variables from method只能从方法访问父类变量
【发布时间】:2016-07-27 19:56:53
【问题描述】:

所以我遇到的问题是这个。我编写了一个方法作为程序的一部分,用于显示 ArrayList 中的所有对象。在这个 ArrayList 中存储了三种不同类型的对象:Fungus、Flowers 和 Weeds。我可以毫无问题地调用plantList.get(i).getName()plantList.get(i).getColor 方法。这两个变量都属于名为 Plant 的父类。然而,当调用下面的方法plantList.get(i).getPoison()(这个方法属于Fungus子类)时,我得到一个编译器错误,说它找不到变量Fungus。

我已经对子类独有的所有其他变量进行了尝试,并且发生了同样的事情。所以我可以从父类“Plant”访问变量,但不能从“Fungus”“Flower”或“Weed”的任何子类访问。我不熟悉使用子类和超类,所以我很难弄清楚问题出在哪里。

public static void displayPlant(ArrayList<Plant> plantList) {
    for (int i = 0; i < plantList.size(); i++) {
        System.out.print(plantList.get(i).getName());
        System.out.print(plantList.get(i).getID());
        System.out.print(plantList.get(i).getColor());
        if (plantList.get(i).contains(Fungus) == true) {
            System.out.print(plantList.get(i).getPoison());
        }
        else if (plantList.get(i).contains(Flower) == true) {
            System.out.print(plantList.get(i).getSmell());
            System.out.print(plantList.get(i).getThorns());
        }
        else {
            System.out.print(plantList.get(i).getPoison());
            System.out.print(plantList.get(i).getEdible());
            System.out.print(plantList.get(i).getMedicinal());
        }
    }
}

【问题讨论】:

  • 这是预期行为。您应该将类​​型转换为相应的子类类型,以便访问子类特定的方法。

标签: java inheritance subclass superclass


【解决方案1】:

一个好的解决方案是使用动态调度。 IE。让元素自己决定要打印什么信息。

class Plant {
    ...
    public String toString() {
        return String.join(" ", getName(), getID(), getColor());
    }
}

class Fungus extends Plant {
    ...
    @Override
    public String toString() {
        return String.join(" ", super.toString(), getPoison());
    }
}

class Flower extends Plant {
    ...
    @Override
    public String toString() {
        return String.join(" ", super.toString(), getSmell(), getThorns());
    }
}

class Weed extends Plant {
    ...
    @Override
    public String toString() {
        return String.join(" ", super.toString(), getPoison(), getEdible(), getMedicinal());
    }
}

您的循环将如下所示:

public static void displayPlant(ArrayList<Plant> plantList) {
    for(Plant p : plantList)
        System.out.println(p); // This calls toString
}

【讨论】:

  • 附言。在toString 中,我使用String.join 创建了一个类似value1 value2 value3 的字符串。但是,当然,您可以使用任何您喜欢的格式返回 String
  • 感谢您的帮助!我需要花一点时间来查看和消化这个解决方案,但它似乎比我做的方式简洁得多。
【解决方案2】:

其实你做错了,

一旦将子类的对象保存到父类变量中,然后在编译时它只会搜索或查找父类中声明的方法和变量。

如果你想访问子类变量那么你需要先找出子类,然后根据它进行类型转换

public static void displayPlant(ArrayList<Plant> plantList) {
    for (int i = 0; i < plantList.size(); i++) {
        Plant plant=plantList.get(i);
        if (plant instanceof Flowers) {
            Flowers fl=(Flowers)plant;
            //TODO do whatever you want to do
        }
        else if(plant instanceof Weeds) {
            Weeds wd=(Weeds)plant;
            //TODO do whatever you want to do
        }else if (plant instanceof Fungus) {
            Fungus wd=(Fungus)plant;
            //TODO do whatever you want to do
        }
    }

希望对你有所帮助。谢谢

【讨论】:

  • 非常感谢!我认为这就是我在结构方面寻找的东西。我将在今晚/明天晚些时候实施它并报告它是如何工作的。
【解决方案3】:

Java 是strongly typed 语言。这意味着,如果你想使用子类的方法,你必须 cast down 子类。

在你的例子中:

...
else if (plantList.get(i) instanceof Fungus) { //check if plant is fungus
    System.out.print(plantList.get(i).getSmell());
    System.out.print(((Fungus)plantList.get(i)).getThorns()); //downcasting
}
...

您使用constains 进行的检查不起作用。要检查对象的类型,您需要使用运算符instanceof

【讨论】:

  • 非常感谢!这解决了我的问题的另一面,即执行子类的命令。非常感谢
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2020-02-13
  • 2012-07-04
  • 1970-01-01
  • 1970-01-01
  • 2011-05-30
  • 2012-10-28
相关资源
最近更新 更多