2012-02-07 88 views
1

我有一个数据库表,其中包含一个名为type的列。对于数据库列中的每一行,我必须根据类型创建一个对象。目前,我使用if else语句为:根据类型创建不同的对象

if (type.equals("object1")){ 
    Object1 object1 = new Object1(); 
} 
else if (type.equals("object2")){ 
    Object2 object2 = new Object2(); 
} 

有些更好的办法是使用一个枚举,作为类型的数量是有限的,但有没有可能让一个对象的创建依赖于价值的字符串?

我愿意以另一种方式解决我的问题,而不是我试图解决的问题。

回答

1

您可以使用

Object o = Class.forName(type).newInstance(); 

如果你有一个int和两个字符串,因为你需要的参数。

Object o = Class.forName(type) 
       .getConstructor(int.class, String.class, String.class) 
       .newInstance(intValue, string1, string2); 

另一种可能性是使用工厂方法

Object o = getClass().getMethod("create_" + type).invoke(null); 

static Object1 create_object1() { 
    return new Object1(/* with args */); 
} 

static Object2 create_object2() { 
    return new Object2(/* with other args */); 
} 

但最灵活的方法可能是使用一个开关

Object o; 
switch(type) { // in Java 7 
    case "object1": o = new Object1(); break; 
    case "object2": o = new Object2(); break; 

什么是Java中8使用闭包更优雅。

http://cr.openjdk.java.net/~briangoetz/lambda/lambda-state-final.html

+1

我正在为Android开发,因此switch语句不起作用 – 2012-02-07 10:22:23

+0

好点,关闭也一样。第一个例子会工作。您可能需要在名称的开头添加一个包,并更改第一个字母的大小写。 ;) – 2012-02-07 10:27:17

+0

我想对象o = Class.forName(type).newInstance();我认为这将最适合我。然而,我坚持传递构造函数变量。我需要通过一个int和两个字符串,这个问题解决这个问题,但我似乎无法弄清楚(或只是了解会发生什么)http://stackoverflow.com/questions/234600/can-i-class-class -newinstance-with-constructor-arguments – 2012-02-07 10:48:13

3

您可以创建一个从StringClass的地图并使用newInstance。然而,这依赖于无参数构造函数的存在。

import java.util.*; 

class Test { 
    public static void main(String[] args) throws Exception { 
     Map<String, Class<?>> classes = new HashMap<String, Class<?>>(); 
     classes.put("object1", Object1.class); 
     classes.put("object2", Object2.class); 

     String type = "object2"; 

     Object obj = classes.get(type).newInstance(); 

     //... 
    } 
} 

class Object1 { ... } 

class Object2 { ... } 
+0

或者所有具有相同参数的构造函数。例如'Object1(Object [] values)' – 2012-02-07 10:14:58

+0

啊,好点。 – aioobe 2012-02-07 10:17:08

+0

我确实想过使用Map,过去我也这样做过。如果你有Java 7,交换机可能更优雅。 – 2012-02-07 10:18:20

0
package main; 

import java.util.ArrayList; 
import java.util.List; 

public class TempClass { 
    public static void main(String[] args) throws ClassNotFoundException, IllegalAccessException, InstantiationException { 
     List<Class> classes = new ArrayList<Class>() {{ 
      add(Class.forName("main.Object1")); 
      add(Class.forName("main.Object2")); 
     }}; 
     for (Class aClass : classes) { 
      Object o = aClass.newInstance(); 
     } 
    } 
} 


package main; 

public class Object1 { 
    public Object1() { 
     System.out.println("Object1"); 
    } 
} 


package main; 

public class Object2 { 
    public Object2() { 
     System.out.println("Object2"); 
    } 
} 
0

使用switchif声明强制你硬编码各种选项,所以你会得到一个不太灵活的解决方案,在我看来。

我总是比较喜欢编写可扩展的代码以用于未来的软件开发。

所以我建议你使用Object o = Class.forName(type).newInstance();在你的数据库中放入你要实例化的类的名字,如果你需要的话,还可以使用自定义标签。

编辑

如果你有一些参数传递到你必须实例化类的构造函数,为方便使用Constructor.newInstance()代替Class.newInstance();和使用Constructor对象实例化的类,例如:

Class myClass = Class.forName(type); 

// get the constructor you need specifying parameters type 
Constructor constructor = myClass.getConstructor(String.class); 

Object myObject = constructor.newInstance("constructor-arg"); 
0

如果有问题的类是继承层次结构的一部分,你碰巧使用某种映射工具/像Hibernate或iBATIS的/ MyBatis的框架,它可能包括一个方法来配置这样CAL带领鉴赏家。

然后,您可以将您的“类型”列配置为descriminator列,并将支持的值映射到您的类。当您从数据库中检索对象时,框架将为您实例化适当的类。