假设我有一个非常简单的java对象,它只有一些getXXX和setXXX属性。此对象仅用于处理值,基本上是记录或类型安全(和性能)映射。我经常需要将此对象转换为键值对(字符串或类型安全)或从键值对转换为此对象。
除了反射或手动编写代码以进行此转换之外,实现此目的的最佳方法是什么?
示例可能是通过jms发送此对象,而不使用ObjectMessage类型(或将传入消息转换为正确类型的对象)。
答案 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负责其余的事情!!!
答案 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作为依赖。
上搜索“Object-to-Map Transformer”本质上,它遍历从作为输入给定的对象可到达的整个对象图,并从对象上的所有基本类型/字符串字段生成映射。它可以配置为输出:
以下是他们页面的一个例子:
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框架:
并利用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,
object
转换为Json 将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]}