如何将Java对象(bean)转换为键值对(反之亦然)?

时间:2009-04-11 07:37:52

标签: java reflection collections javabeans

假设我有一个非常简单的java对象,它只有一些getXXX和setXXX属性。此对象仅用于处理值,基本上是记录或类型安全(和性能)映射。我经常需要将此对象转换为键值对(字符串或类型安全)或从键值对转换为此对象。

除了反射或手动编写代码以进行此转换之外,实现此目的的最佳方法是什么?

示例可能是通过jms发送此对象,而不使用ObjectMessage类型(或将传入消息转换为正确类型的对象)。

23 个答案:

答案 0 :(得分:170)

很多潜在的解决方案,但让我们再添加一个。使用Jackson(JSON处理库)进行“json-less”转换,如:

ObjectMapper m = new ObjectMapper();
Map<String,Object> props = m.convertValue(myBean, Map.class);
MyBean anotherBean = m.convertValue(props, MyBean.class);

this blog entry还有更多例子)

你基本上可以转换任何兼容的类型:兼容意味着如果你确实从类型转换为JSON,从那个JSON转换为结果类型,条目将匹配(如果配置正确也可以忽略无法识别的那些)。

适用于人们期望的案例,包括地图,列表,数组,基元,类似bean的POJO。

答案 1 :(得分:50)

总有apache commons beanutils但当然它在引擎盖下使用反射

答案 2 :(得分:8)

代码生成将是我能想到的唯一其他方式。就个人而言,我有一个通常可重复使用的反射解决方案(除非该部分代码绝对是性能关键)。使用JMS听起来像是矫枉过正(额外的依赖,这甚至不是它的意思)。此外,它可能也会在引擎盖下使用反射。

答案 3 :(得分:7)

这是一种将Java对象转换为Map

的方法
public static Map<String, Object> ConvertObjectToMap(Object obj) throws 
    IllegalAccessException, 
    IllegalArgumentException, 
    InvocationTargetException {
        Class<?> pomclass = obj.getClass();
        pomclass = obj.getClass();
        Method[] methods = obj.getClass().getMethods();


        Map<String, Object> map = new HashMap<String, Object>();
        for (Method m : methods) {
           if (m.getName().startsWith("get") && !m.getName().startsWith("getClass")) {
              Object value = (Object) m.invoke(obj);
              map.put(m.getName().substring(3), (Object) value);
           }
        }
    return map;
}

这是如何称呼它

   Test test = new Test()
   Map<String, Object> map = ConvertObjectToMap(test);

答案 4 :(得分:4)

使用Java 8,您可以尝试这样做:

public Map<String, Object> toKeyValuePairs(Object instance) {
    return Arrays.stream(Bean.class.getDeclaredMethods())
            .collect(Collectors.toMap(
                    Method::getName,
                    m -> {
                        try {
                            Object result = m.invoke(instance);
                            return result != null ? result : "";
                        } catch (Exception e) {
                            return "";
                        }
                    }));
}

答案 5 :(得分:3)

JSON,例如使用XStream + Jettison,是一个带键值对的简单文本格式。例如,Apache ActiveMQ JMS消息代理支持它与其他平台/语言的Java对象交换。

答案 6 :(得分:3)

最好的解决方案是使用Dozer。你只需要在mapper文件中输入这样的东西:

<mapping map-id="myTestMapping">
  <class-a>org.dozer.vo.map.SomeComplexType</class-a>
  <class-b>java.util.Map</class-b>
</mapping> 

就是这样,Dozer负责其余的事情!!!

Dozer Documentation URL

答案 7 :(得分:3)

只需使用反射和Groovy:

def Map toMap(object) {             
return object?.properties.findAll{ (it.key != 'class') }.collectEntries {
            it.value == null || it.value instanceof Serializable ? [it.key, it.value] : [it.key,   toMap(it.value)]
    }   
}

def toObject(map, obj) {        
    map.each {
        def field = obj.class.getDeclaredField(it.key)
        if (it.value != null) {
            if (field.getType().equals(it.value.class)){
                obj."$it.key" = it.value
            }else if (it.value instanceof Map){
                def objectFieldValue = obj."$it.key"
                def fieldValue = (objectFieldValue == null) ? field.getType().newInstance() : objectFieldValue
                obj."$it.key" = toObject(it.value,fieldValue) 
            }
        }
    }
    return obj;
}

答案 8 :(得分:3)

使用juffrou-reflect的BeanWrapper。它非常高效。

以下是将bean转换为地图的方法:

public static Map<String, Object> getBeanMap(Object bean) {
    Map<String, Object> beanMap = new HashMap<String, Object>();
    BeanWrapper beanWrapper = new BeanWrapper(BeanWrapperContext.create(bean.getClass()));
    for(String propertyName : beanWrapper.getPropertyNames())
        beanMap.put(propertyName, beanWrapper.getValue(propertyName));
    return beanMap;
}
我自己开发了Juffrou。它是开源的,因此您可以自由使用它并进行修改。如果您对此有任何疑问,我将非常乐意回复。

干杯

卡洛斯

答案 9 :(得分:3)

使用Spring时,还可以使用Spring Integration对象到映射变换器。可能不值得为此添加Spring作为依赖。

有关文档,请在http://docs.spring.io/spring-integration/docs/4.0.4.RELEASE/reference/html/messaging-transformation-chapter.html

上搜索“Object-to-Map Transformer”

本质上,它遍历从作为输入给定的对象可到达的整个对象图,并从对象上的所有基本类型/字符串字段生成映射。它可以配置为输出:

  • 平面地图:{rootObject.someField = Joe,rootObject.leafObject.someField = Jane}或
  • 结构化地图:{someField = Joe,leafObject = {someField = Jane}}。

以下是他们页面的一个例子:

public class Parent{
    private Child child;
    private String name; 
    // setters and getters are omitted
}

public class Child{
   private String name; 
   private List<String> nickNames;
   // setters and getters are omitted
}

输出将是:

  

{person.name = George,person.child.name = Jenna,   person.child.nickNames [0] =宾博。 。 。等}

也可以使用反向变压器。

答案 10 :(得分:2)

您可以使用Joda框架:

http://joda.sourceforge.net/

并利用JodaProperties。这确实规定您以特定方式创建bean,并实现特定接口。但是,它允许您从特定类返回属性映射,而不进行反射。示例代码在这里:

http://pbin.oogly.co.uk/listings/viewlistingdetail/0e78eb6c76d071b4e22bbcac748c57

答案 11 :(得分:2)

如果您不想对每个getter和setter进行硬编码调用,那么反射是调用这些方法的唯一方法(但并不难)。

你可以重构有问题的类来使用Properties对象来保存实际数据,让每个getter和setter只调用get / set吗?然后你有一个非常适合你想做的结构。甚至还有以键值形式保存和加载它们的方法。

答案 12 :(得分:2)

您可以使用java 8流过滤器收集器属性

public Map<String, Object> objectToMap(Object obj) {
    return Arrays.stream(YourBean.class.getDeclaredMethods())
            .filter(p -> !p.getName().startsWith("set"))
            .filter(p -> !p.getName().startsWith("getClass"))
            .filter(p -> !p.getName().startsWith("setClass"))
            .collect(Collectors.toMap(
                    d -> d.getName().substring(3),
                    m -> {
                        try {
                            Object result = m.invoke(obj);
                            return result;
                        } catch (Exception e) {
                            return "";
                        }
                    }, (p1, p2) -> p1)
            );
}

答案 13 :(得分:2)

当然有绝对最简单的转换方式 - 根本没有转换!

而不是使用类中定义的私有变量,使该类只包含一个存储实例值的HashMap。

然后你的getter和setter返回并设置值进出HashMap,当它需要将它转换为地图时,瞧! - 它已经是一张地图。

通过一点AOP魔法,您甚至可以通过允许仍然使用特定于每个值名称的getter和setter来保持bean固有的不灵活性,而无需实际编写单独的getter和setter。

答案 14 :(得分:1)

可能迟到了。您可以使用Jackson并将其转换为Properties对象。这适用于嵌套类,如果你想要a.b.c = value中的键。

SerializationConfig config = mapper.getSerializationConfig()
                .withRootName("suffix");
mapper.setConfig(config);

如果你想要一些后缀,那就做吧

<dependency>
  <groupId>com.fasterxml.jackson.dataformat</groupId>
  <artifactId>jackson-dataformat-properties</artifactId>
</dependency>

需要添加此依赖项

{{1}}

答案 15 :(得分:1)

在Jackson库的帮助下,我能够找到String / integer / double类型的所有类属性,以及Map类中的相应值。 (不使用反射api!

TestClass testObject = new TestClass();
com.fasterxml.jackson.databind.ObjectMapper m = new com.fasterxml.jackson.databind.ObjectMapper();

Map<String,Object> props = m.convertValue(testObject, Map.class);

for(Map.Entry<String, Object> entry : props.entrySet()){
    if(entry.getValue() instanceof String || entry.getValue() instanceof Integer || entry.getValue() instanceof Double){
        System.out.println(entry.getKey() + "-->" + entry.getValue());
    }
}

答案 16 :(得分:1)

如果您真的想要性能,可以使用代码生成路径。

你可以通过自己的反思和构建一个混合的AspectJ ITD来实现这一点。

或者您可以使用Spring Roo并制作Spring Roo Addon。您的Roo插件将执行与上述类似的操作,但是对于使用Spring Roo并且您不必使用Runtime Annotations的所有人都可以使用。

我做过两件事。人们对Spring Roo不屑一顾,但它确实是Java最全面的代码生成。

答案 17 :(得分:1)

另一种可能的方式就在这里。

BeanWrapper提供设置和获取属性值的功能(单独使用  或者批量处理),获取属性描述符,并查询属性以确定它们是否存在  可读或可写。

Company c = new Company();
 BeanWrapper bwComp = BeanWrapperImpl(c);
 bwComp.setPropertyValue("name", "your Company");

答案 18 :(得分:1)

如果涉及到键值列表映射的简单对象树,其中key可能是从对象的根元素到正在检查的叶子的虚线路径描述,则很明显树转换为键值列表是与xml映射的对象相当。 XML文档中的每个元素都有一个已定义的位置,可以转换为路径。因此,我将XStream作为基本且稳定的转换工具,并使用自己的实现替换了分层驱动程序和编写器部分。 XStream还带有一个基本的路径跟踪机制 - 与其他两个机制相结合 - 严格地导致适合该任务的解决方案。

答案 19 :(得分:1)

你应该写一个通用的转换服务!使用泛型保持其自由类型(因此您可以将每个对象转换为key =&gt;值并返回)。

什么字段应该是关键?从bean中获取该字段,并将任何其他非瞬态值附加到值映射中。

回来很简单。读取密钥(x)并首先写入密钥,然后将每个列表条目写回新对象。

您可以使用apache commons beanutils获取bean的属性名称!

答案 20 :(得分:1)

我的JavaDude Bean Annotation Processor生成代码来执行此操作。

http://javadude.googlecode.com

例如:

@Bean(
  createPropertyMap=true,
  properties={
    @Property(name="name"),
    @Property(name="phone", bound=true),
    @Property(name="friend", type=Person.class, kind=PropertyKind.LIST)
  }
)
public class Person extends PersonGen {}

上面生成了包含createPropertyMap()方法的超类PersonGen,该方法为使用@Bean定义的所有属性生成Map。

(请注意,我正在为下一个版本略微更改API - 注释属性将是defineCreatePropertyMap = true)

答案 21 :(得分:0)

通过使用Gson,

  1. 将POJO object转换为Json
  2. 将Json转换为Map

        retMap = new Gson().fromJson(new Gson().toJson(object), 
                new TypeToken<HashMap<String, Object>>() {}.getType()
        );
    

答案 22 :(得分:0)

我们可以使用Jackson库将Java对象轻松转换为Map。

<dependency>
   <groupId>com.fasterxml.jackson.core</groupId>
   <artifactId>jackson-databind</artifactId>
   <version>2.6.3</version>
</dependency>

如果在Android项目中使用,则可以按以下步骤在应用的build.gradle中添加杰克逊:

implementation 'com.fasterxml.jackson.core:jackson-core:2.9.8'
implementation 'com.fasterxml.jackson.core:jackson-annotations:2.9.8'
implementation 'com.fasterxml.jackson.core:jackson-databind:2.9.8'

示例实现

public class Employee {

    private String name;
    private int id;
    private List<String> skillSet;

    // getters setters
}

public class ObjectToMap {

 public static void main(String[] args) {

    ObjectMapper objectMapper = new ObjectMapper();

    Employee emp = new Employee();
    emp.setName("XYZ");
    emp.setId(1011);
    emp.setSkillSet(Arrays.asList("python","java"));

    // object -> Map
    Map<String, Object> map = objectMapper.convertValue(emp, 
    Map.class);
    System.out.println(map);

 }

}

输出:

{name = XYZ,id = 1011,技能= [python,java]}