Java:使用对象名称的String输入创建对象

时间:2013-12-01 03:42:33

标签: java object

我有两个班,狗和猫:

class Dog{
   public void speak(){
        System.out.print("Woof!");
   }
}

class Cat{
   public void speak(){
       System.out.print("Meow!");
   }
}

在我的主要内容中,我将名称命名为String,“Cat”或“Dog”。

public static void main(String [] args){
    Scanner sc = new Scanner(System.in);
    String name = sc.next();

    Class<?> cls = Class.forName(name);
    Object object = cls.newInstance();
}

但现在我希望能够将方法称为“说话”。但是我必须将对象强制转换为猫或狗,因为“对象”显然没有内置“说话”方法。所以我的解决方案是创建另一个类(我不能使用if-statements btw):

class Animal{
    public void speak(){
    }
}

然后“Cat”和“Dog”都可以扩展Animal并覆盖其方法。有没有其他方法可以做到这一点没有使用另一个方法/使用if语句? (包括开关盒,三元操作器)。提前谢谢。

另一个问题: 如果我将METHOD的名称作为输入,我该如何调用它?例如:

class Dog{
    public void speak(){}
    public void bark(){}
}

如果我以“说”或“吠叫”的形式接受字符串,如何在不使用if语句的情况下调用该方法?

4 个答案:

答案 0 :(得分:2)

您可以使用Class.getMethodMethod.invoke进行反思。

尽管如此,创建Animal类确实是最干净的方法。什么阻止你做这件事?

答案 1 :(得分:0)

你走在正确的轨道上。最简单的方法是创建一个动物类,让狗和猫从它继承并使它们都实现自己的speak()版本。有没有理由你不想创建另一个类?

答案 2 :(得分:0)

好的,这是按优先顺序排列的两种方法:

abstract class Animal {
    public abstract void speak();
}

class Dog extends Animal {
    @Override
    public void speak() {
        System.out.println("Woof woof");
    }
}

class Cat extends Animal {
    @Override
    public void speak() {
        System.out.println("Miauw");
    }
}

public static void main(String[] args) {
    String type = "Dog";
    Class<?> clazz;
    try {
        clazz = Class.forName(type);
        Animal pet = (Animal) clazz.newInstance();
        pet.speak();
    } catch (Exception e) {
        e.printStackTrace();
    }
}

我正在使用基类,因为可以假设Animal将包含每个动物共享的更多字段(名称,种类等)。如果不是这种情况,那么你应该去一个界面。

或反思:

public class Test {
    public static void main(String[] args) {
        String type = "Dog";
        Class<?> clazz;
        try {
            clazz = Class.forName(type);
            for(Method method : clazz.getMethods()){
                if(method.getName().equals("speak")){
                    method.invoke(clazz.newInstance(), null);
                }
            }
        } catch (CException e) {
            e.printStackTrace();
        }
    }
}

答案 3 :(得分:0)

您无需创建class动物 - 创建interface

interface Animal{
    public void speak();
}

CatDog实现它。然后在main()

Class<?> cls = Class.forName(name);
Animal animal = cls.newInstance();
animal.speak();

无需投射或使用if / else。

使用继承/抽象类的唯一原因是当您想要重用功能时(实现一次方法并在几个类中使用它)。否则 - 更好地使用接口。

至于方法名称,如果你想要“明智的”解决方案:使用switch(从Java 7支持)。否则,请参阅@ immibis的回答。