Java反射:创建一个实现类

时间:2009-07-04 19:41:32

标签: java reflection interface

Class someInterface = Class.fromName("some.package.SomeInterface");

我现在如何创建一个实现someInterface的新类?

我需要创建一个新类,并将其传递给需要SomeInterface作为参数的函数。

4 个答案:

答案 0 :(得分:72)

轻松,java.lang.reflect.Proxy救援!

完整的工作示例

interface IRobot {

    String Name();

    String Name(String title);

    void Talk();

    void Talk(String stuff);

    void Talk(int stuff);

    void Talk(String stuff, int more_stuff);

    void Talk(int stuff, int more_stuff);

    void Talk(int stuff, String more_stuff);
}

public class ProxyTest {
    public static void main(String args[]) {
        IRobot robot = (IRobot) java.lang.reflect.Proxy.newProxyInstance(
                IRobot.class.getClassLoader(),
                new java.lang.Class[] { IRobot.class },
                new java.lang.reflect.InvocationHandler() {

            @Override
            public Object invoke(Object proxy, java.lang.reflect.Method method, Object[] args) throws java.lang.Throwable {
                String method_name = method.getName();
                Class<?>[] classes = method.getParameterTypes();

                if (method_name.equals("Name")) {
                    if (args == null) {
                        return "Mr IRobot";
                    } else {
                        return args[0] + " IRobot";
                    }
                } else if (method_name.equals("Talk")) {
                    switch (classes.length) {
                        case 0:
                            System.out.println("Hello");
                            break;
                        case 1:
                            if (classes[0] == int.class) {
                                System.out.println("Hi. Int: " + args[0]);
                            } else {
                                System.out.println("Hi. String: " + args[0]);
                            }
                            break;
                        case 2:
                            if (classes[0] == String.class) {
                                System.out.println("Hi. String: " + args[0] + ". Int: " + args[1]);
                            } else {
                                if (classes[1] == String.class) {
                                    System.out.println("Hi. int: " + args[0] + ". String: " + args[1]);
                                } else {
                                    System.out.println("Hi. int: " + args[0] + ". Int: " + args[1]);
                                }
                            }
                            break;
                    }
                }
                return null;
            }
        });

        System.out.println(robot.Name());
        System.out.println(robot.Name("Dr"));
        robot.Talk();
        robot.Talk("stuff");
        robot.Talk(100);
        robot.Talk("stuff", 200);
        robot.Talk(300, 400);
        robot.Talk(500, "stuff");
    }
}

答案 1 :(得分:52)

创建一些假装即时实现接口的东西实际上并不太难。实施java.lang.reflect.Proxy后,您可以使用InvocationHandler来处理任何方法调用。

当然,您实际上可以使用BCEL等库来生成真正的类。

如果这是出于测试目的,你应该看看像jMockEasyMock这样的模拟框架。

答案 2 :(得分:3)

如果您想超越界面,可能需要查看cglibobjenesis。它们将共同允许您执行一些非常强大的功能,扩展抽象类并实例化它。 (例如,jMock将它们用于此目的。)

如果您想坚持使用接口,请执行Jon Skeet所说的内容:)。

答案 3 :(得分:-3)

实际上,您必须在Class.fromName()方法中使用类名并强制转换为您的接口类型。看看下面的示例是否有帮助。

public class Main {

    public static void main(String[] args) throws Exception {
        Car ferrari = (Car) Class.forName("Mercedez").newInstance();
        System.out.println(ferrari.getName());
    }
}

interface Car {
    String getName();
}

class Mercedez implements Car {

    @Override
    public String getName() {
        return "Mercedez";
    }

}

class Ferrari implements Car {

    @Override
    public String getName() {
        return "Ferrari";
    }

}