在Java中使用反射创建一个新实例,并将引用变量类型设置为新实例类名?

问题描述 投票:0回答:7

我查看的所有反射示例都显示创建未知实现的新实例,并将该实现转换为其接口。问题在于,现在您无法在实现类上调用任何新方法(只能覆盖),因为您的对象引用变量具有接口类型。这是我所拥有的:

Class c = null;
try {
    c = Class.forName("com.path.to.ImplementationType");
} catch (ClassNotFoundException e) {
    e.printStackTrace();
}
InterfaceType interfaceType = null;
try {
    interfaceType = (InterfaceType)c.newInstance();
} catch (InstantiationException e) {
    e.printStackTrace();
} catch (IllegalAccessException e) {
    e.printStackTrace();
}

如果我只引用“com.path.to.ImplementationType”,并且我不知道该类型可能是什么(它来自配置文件),那么我如何使用类名来转换它到实现类型?这可能吗?

java reflection
7个回答
33
投票

这句话似乎概括了您问题的症结:

问题在于,现在您无法在实现类上调用任何新方法(只能重写),因为您的对象引用变量具有接口类型。

您非常陷入当前的实现中,因为您不仅必须尝试强制转换,还需要定义要在此子类上调用的方法。我看到两个选项:

1. 如其他地方所述,您不能使用类名称的字符串表示形式将反射实例转换为已知类型。但是,您可以使用

String
equals()
测试来确定您的类是否属于您想要的类型,然后执行硬编码转换:

try {
   String className = "com.path.to.ImplementationType";// really passed in from config
   Class c = Class.forName(className);
   InterfaceType interfaceType = (InterfaceType)c.newInstance();
   if (className.equals("com.path.to.ImplementationType") {
      ((ImplementationType)interfaceType).doSomethingOnlyICanDo();
   } 
} catch (Exception e) {
   e.printStackTrace();
}

这看起来非常丑陋,并且它破坏了您所拥有的良好的配置驱动流程。我不建议你这样做,这只是一个例子。

2. 您的另一个选择是将反射从

Class
/
Object
创建扩展到包括
Method
反射。如果您可以从配置文件传入的字符串创建
Class
,您还可以从该配置文件传入方法名称,并通过反射从
Method
对象获取
Class
本身的实例。然后您可以调用
invoke
(http://java.sun.com/javase/6/docs/api/java/lang/reflect/Method.html#invoke(java.lang.Object, java.lang. Object...)) 在
Method
上,传入您创建的类的实例。我想这会帮助你得到你想要的东西。

这里有一些代码作为示例。请注意,我冒昧地对方法的参数进行了硬编码。您也可以在配置中指定它们,并且需要反映它们的类名来定义它们的

Class
对象和实例。

public class Foo {

    public void printAMessage() {
    System.out.println(toString()+":a message");
    }
    public void printAnotherMessage(String theString) {
        System.out.println(toString()+":another message:" + theString);
    }

    public static void main(String[] args) {
        Class c = null;
        try {
            c = Class.forName("Foo");
            Method method1 = c.getDeclaredMethod("printAMessage", new Class[]{});
            Method method2 = c.getDeclaredMethod("printAnotherMessage", new Class[]{String.class});
            Object o = c.newInstance();
            System.out.println("this is my instance:" + o.toString());
            method1.invoke(o);
            method2.invoke(o, "this is my message, from a config file, of course");
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        } catch (NoSuchMethodException nsme){
            nsme.printStackTrace();
        } catch (IllegalAccessException iae) {
            iae.printStackTrace();
        } catch (InstantiationException ie) {
            ie.printStackTrace();
        } catch (InvocationTargetException ite) {
            ite.printStackTrace();
        }
    }
}

和我的输出:

this is my instance:Foo@e0cf70
Foo@e0cf70:a message
Foo@e0cf70:another message:this is my message, from a config file, of course

5
投票
//====Single Class Reference used to retrieve object for fields and initial values. Performance enhancing only====          
Class<?>    reference           =   vector.get(0).getClass();
Object      obj                 =   reference.newInstance();
Field[]     objFields           =   obj.getClass().getFields(); 

3
投票

我不确定我是否正确地回答了你的问题,但你似乎想要这样的东西:

    Class c = null;
    try {
        c = Class.forName("com.path.to.ImplementationType");
    } catch (ClassNotFoundException e) {
        e.printStackTrace();
    }
    T interfaceType = null;
    try {
        interfaceType = (T) c.newInstance();
    } catch (InstantiationException e) {
        e.printStackTrace();
    } catch (IllegalAccessException e) {
        e.printStackTrace();
    }

其中 T 可以在方法级别或类级别定义,即

<T extends InterfaceType>


3
投票

作为 akf 答案的附录,您可以使用 instanceof 检查而不是 String equals() 调用:

String cname="com.some.vendor.Impl";
try {
  Class c=this.getClass().getClassLoader().loadClass(cname);
  Object o= c.newInstance();
  if(o instanceof Spam) {
    Spam spam=(Spam) o;
    process(spam);
  }
  else if(o instanceof Ham) {
    Ham ham = (Ham) o;
    process(ham);
  }
  /* etcetera */
}
catch(SecurityException se) {
  System.err.printf("Someone trying to game the system?%nOr a rename is in order because this JVM doesn't feel comfortable with: “%s”", cname);
  se.printStackTrace();
}
catch(LinkageError le) {
  System.err.printf("Seems like a bad class to this JVM: “%s”.", cname);
  le.printStackTrace();
}
catch(RuntimeException re) { 
  // runtime exceptions I might have forgotten. Classloaders are wont to produce those.
  re.printStackTrace();
}
catch(Exception e) { 
  e.printStackTrace();
}

注意某些值的自由硬编码。无论如何,要点是:

  1. 使用instanceof而不是equals()。如果有的话,重构时它会更好地配合。
  2. 一定要捕获这些运行时错误和安全错误。

2
投票

您希望能够传入一个类并获取该类的类型安全实例吗?尝试以下操作:

public static void main(String [] args) throws Exception {
    String s = instanceOf(String.class);
}

public static <T> T instanceOf (Class<T> clazz) throws Exception {
    return clazz.newInstance();
}

0
投票

如果您知道

Class
ImplementationType
,您可以创建它的实例。所以你想做的事是不可能的。


0
投票

按照 Java 指南。

this.getClass().getDeclaredConstructor().newInstance()

© www.soinside.com 2019 - 2024. All rights reserved.