【问题标题】:Java method returning HashMap<key, Object> with dynamic objectJava 方法返回带有动态对象的 HashMap<key, Object>
【发布时间】:2014-09-18 19:37:46
【问题描述】:

我有几个 TSV 文件,想读取它们并填充到

的 HashMap 中

[第一个字段 --> 对象中的其他字段]。

为简单起见,假设有两个文件:

文件 1 包含两个字段(字段 1 和字段 2)。
文件 2 包含三个字段(f1、f2 和 f3)。

所以我定义了两个类,它们的对象是 hashMap 中的值:

Class1{
    String field1 = "";
    String field2 = "";
}
Class2{
    String f1 = "";
    String f2 = "";
    String f3 = "";
}

现在,我有这些方法:

public static HashMap<String, Class1> readTSV1(String fileName, Class1 c){
...
}

public static HashMap<String, Class2> readTSV2(String fileName, Class2 c){
...
}
...

但我不想定义从不同文件读取的各种方法:

我想要这样的东西:

public static HashMap<String, Object> readTSV(String fileName, Class<?> c){
    HashMap<String, c.getClass()> hm = new HashMap<String, c.getClass()>(); //error.
    //Look which field names are in type c, 
    //and then read two or three fields from file, 
    //and put them as key and values of hm (first field is key, other fields are put in instance of the respective class, and put as values)
    return hm;
}

static void main(String[] args){
    Class1 c1;
    HashMap<String, Class1> hm1 = new HashMap<String, Class1>();
    hm1 = readTSV("firstFile.tsv", c1.getClass())

    Class2 c2;
    HashMap<String, Class2> hm1 = new HashMap<String, Class2>();
    hm1 = readTSV("firstFile.tsv", c2.getClass())

    ...
}

有什么想法吗? ...

【问题讨论】:

  • 听起来你真正希望你的方法返回的是HashMap&lt;String,List&lt;String&gt;&gt;。您可以将int 传递给您的readTSV 方法来说明要读取多少个字段。或者您可以继续阅读字段并将它们添加到行尾。
  • 它有效,但如果适用的话,这个更易读。例如,在 main 中,我们可以说:(Class1)hm1.get("field1").field1 而不是 fields[i]。文件数量及其不同的字段太多,可能会导致错误。
  • 好吧,要完全按照您的描述进行操作,您需要反思,查看您班级中的字段。相信我,添加反射代码后它的可读性会大大降低。
  • 我不明白@Alisa 的效果如何。你能举一个这样的TSV文件的例子吗?以及您实际上想如何使用它?

标签: java class object hashmap


【解决方案1】:

使用Hashmap&lt;String, List&lt;String&gt;&gt; 可能是最简单的方法。但是,如果您真的想要在对象中使用这些,您可以使用接口做一些事情。

public interface CSVConvertable {
    /* sets values in this class according to a row in the CSV file */
    public void setCSVValues(String [] values);
}

class Class1 implements CSVConvertable {
    String field1 = "";
    String field2 = "";
    @Override
    public void setCSVValues(String[] values) {
        field1 = values[0];
        field2 = values[1];
    }
}
class Class2 implements CSVConvertable {
    String f1 = "";
    String f2 = "";
    String f3 = "";
    @Override
    public void setCSVValues(String[] values) {
        f1 = values[0];
        f2 = values[1];
        f3 = values[2];
    }
}

public static <T extends CSVConvertable> HashMap<String, T> readTSV(String fileName, Class<T> c) throws InstantiationException, IllegalAccessException{
    HashMap<String, T> hm = new HashMap<String, T>();
    while(/* read rows in csv*/) {
        CSVConvertable conv = c.newInstance();
        conv.setCSVValues(/*your row array here*/);
    }

    return hm;
}


static void main(String[] args){
    HashMap<String, Class1> hm1 = new HashMap<String, Class1>();
    hm1 = readTSV("firstFile.tsv", Class1.class);

    HashMap<String, Class2> hm2 = new HashMap<String, Class2>();
    hm2 = readTSV("firstFile.tsv", Class2.class);

    ...
}

反思

如果你真的想使用反射,这里有一个基本的实现。但是您应该注意,如果您向类中添加了新属性、更改了属性名称或使该类扩展了另一个类,则此实现会发生变化。

public static <T> List<T> readTSV(String fileName, Class<T> c) throws InstantiationException, IllegalAccessException, IntrospectionException, NoSuchFieldException, SecurityException, NoSuchMethodException, IllegalArgumentException, InvocationTargetException{
    List<T> list = new ArrayList<T>(); //error.
    List<String> properties = getBeanProperties(c);
    Collections.sort(properties);

    // loop through all rows of the TSV and set each value
    while(/*read rows in tsv*/) {
        T obj = c.newInstance();
        for(int i=0;i<properties.size();i++) {
            setProperty(obj, properties.get(i), /* get row column [i] */);
        }
        list.add(obj);
    }

    return list;
}


public static void main(String[] args) throws InstantiationException, IllegalAccessException, IntrospectionException, NoSuchFieldException, SecurityException, NoSuchMethodException, IllegalArgumentException, InvocationTargetException{
    List<Class1> hm1 = readTSV("firstFile.tsv", Class1.class);
    System.out.println(hm1);

    List<Class2> hm2 = readTSV("firstFile.tsv", Class2.class);
    System.out.println(hm2);

}

public static void setProperty(Object obj, String propName, Object value) throws NoSuchFieldException, SecurityException, NoSuchMethodException, IllegalAccessException, IllegalArgumentException, InvocationTargetException {
    String setterName = "set" + propName.substring(0, 1).toUpperCase()
            + propName.substring(1);

    Field field = obj.getClass().getDeclaredField(propName);
    if(Modifier.isPrivate(field.getModifiers())) {
        Method method = obj.getClass().getMethod(setterName, field.getType());
        method.invoke(obj, value);
    } else {
        field.set(obj, value);
    }
}

public static List<String> getBeanProperties(Class<?> cl) {
    List<String> properties = new ArrayList<String>();
    // check all declared fields
    for (Field field : cl.getDeclaredFields()) {
        // if field is private then look for setters/getters
        if (Modifier.isPrivate(field.getModifiers())) {
            // changing 1st letter to upper case
            String name = field.getName();
            String upperCaseName = name.substring(0, 1).toUpperCase()
                    + name.substring(1);
            // and have getter and setter
            try {
                String simpleType = field.getType().getSimpleName();
                //for boolean property methods should be isProperty and setProperty(propertyType)
                if (simpleType.equals("Boolean") || simpleType.equals("boolean")) {
                    if ((cl.getDeclaredMethod("is" + upperCaseName) != null)
                            && (cl.getDeclaredMethod("set" + upperCaseName,
                                    field.getType()) != null)) {
                    }
                    properties.add(name);
                } 
                //for not boolean property methods should be getProperty and setProperty(propertyType)
                else {
                    if ((cl.getDeclaredMethod("get" + upperCaseName) != null)
                            && (cl.getDeclaredMethod("set" + upperCaseName,
                                    field.getType()) != null)) {
                    }
                    properties.add(name);
                }
            } catch (NoSuchMethodException | SecurityException e) {
                // if there is no method nothing bad will happen
            }
        } else {
            // Accessible property that isnt defined by the jre
            if(!field.isSynthetic()) {
                properties.add(field.getName());
            }
        }
    }
    return properties;
}

【讨论】:

  • 谢谢。它确实有效,但我希望“Class1 实现 CSVConvertable”类自动执行此任务。我想动态分配类的字段(不提及他们的名字)。让我说类中的字段名称按字母顺序排列(并且与 TSV 的顺序相同)。那么我们可以通过 getNextProperty().value() 之类的方法访问它们吗?
  • @Alisa 您可以通过反射和按名称对属性进行排序来做到这一点,但您不能按它们在源中声明的顺序对它们进行排序。这是因为java是一种编译语言。因此,如果您不想要该界面,IMO 的下一个最佳选择是使用注释。如果您甚至不想使用注释,您可以使用一些反射来获取所有字段名称。我强烈建议不要使用反射来获取属性,除非你真的知道你在做什么。
  • @Alisa 我添加了一个反射实现的例子
【解决方案2】:

你可以使用继承

标记界面

public interface ForClasses{
}

Class1 implements ForClasses {
    ...
}
Class2 implements ForClasses{
    ...
}

那么你可以这样做:

 HashMap<String, ForClasses > hm = new HashMap<String, ForClasses>();

hm 可以在 map 值部分同时保存 class1 对象和 class2 对象....

【讨论】:

  • 但要真正使用它们,你需要强制转换......我更喜欢大卫的方法。
  • 没关系,但是寻找类的字段名称(从文件中读取)呢?我的意思是方法中的三个注释行。
猜你喜欢
  • 2019-03-14
  • 2018-05-20
  • 2013-03-05
  • 1970-01-01
  • 2016-08-20
  • 2016-03-04
  • 1970-01-01
  • 2015-05-24
  • 1970-01-01
相关资源
最近更新 更多