`
allenBen
  • 浏览: 11783 次
  • 性别: Icon_minigender_1
  • 来自: 重庆
文章分类
社区版块
存档分类
最新评论

处理XStream不支持属性、值同时出现的缺陷

阅读更多

XStream在处理<Field name="value">text</Field>的情况无能为力,如果要他支持就要写一个Converter,贴上Converter程序如下:

 

package stream;

import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;

import com.thoughtworks.xstream.converters.Converter;
import com.thoughtworks.xstream.converters.MarshallingContext;
import com.thoughtworks.xstream.converters.UnmarshallingContext;
import com.thoughtworks.xstream.converters.reflection.ObjectAccessException;
import com.thoughtworks.xstream.io.HierarchicalStreamReader;
import com.thoughtworks.xstream.io.HierarchicalStreamWriter;
import com.thoughtworks.xstream.mapper.Mapper;

/**
 * 
 * @author Zhao.Allen
 * 
 */
public class AttributeValueConveter implements Converter {

	public String textName = "value";
	private Mapper mapper;
	private List convertClass = new ArrayList();

	public AttributeValueConveter(Mapper mapper) {
		this.mapper = mapper;
	}

	public void marshal(Object obj, HierarchicalStreamWriter wt, MarshallingContext context) {
		Field[] values = obj.getClass().getDeclaredFields();
		for (int i = 0; i < values.length; i++) {
			Field value = values[i];
			String name = value.getName();
			if (name.equals(textName)) {
				continue;
			}
			try {
				if (!value.isAccessible()) {
					value.setAccessible(true);
				}
				Object v = value.get(obj);
				if (v == null) {
					continue;
				}
				if (Number.class.isAssignableFrom(v.getClass()) || v.getClass().isPrimitive()) {
					wt.addAttribute(mapper.serializedMember(obj.getClass(), name), String.valueOf(v));
				} else {
					wt.addAttribute(mapper.serializedMember(obj.getClass(), name), v.toString());
				}
			} catch (Exception e) {
				throw new ObjectAccessException("Cannot set Field " + value.getName() + obj.getClass(), e);
			}
		}
		try{
			Field field = obj.getClass().getDeclaredField(textName);
			if(!field.isAccessible()){
				field.setAccessible(true);
			}
			Object v = field.get(obj);
			if (null != v && !"".equals(v.toString())) {
				wt.setValue(v.toString());
			}
		} catch (Exception e) {
			throw new ObjectAccessException("Cannot set Field " + textName + obj.getClass(), e);
		}
	}

	public Object unmarshal(HierarchicalStreamReader reader, UnmarshallingContext context) {
		Object obj = context.currentObject();
		if (obj == null) {
			try {
				obj = context.getRequiredType().newInstance();
			} catch (Exception e) {
				throw new ObjectAccessException("Cannot construct " + context.getRequiredType().getName(), e);
			}
		}

		Iterator attNames = reader.getAttributeNames();
		while (attNames.hasNext()) {
			String attName = (String) attNames.next();
			if (attName.equals(textName)) {
				continue;
			}

			try {
				Field field = obj.getClass().getDeclaredField(mapper.realMember(obj.getClass(), attName));
				if (!field.isAccessible()) {
					field.setAccessible(true);
				}
				String v = reader.getAttribute(attName);
				if (null == v || "".equals(v)) {
					continue;
				}
				Class fieldType = field.getType();
				Constructor strnum = fieldType.getDeclaredConstructor(String.class);
				field.set(obj, strnum.newInstance(v));
			} catch (Exception e) {
				e.printStackTrace();
				throw new ObjectAccessException("Cannot construct " + obj.getClass(), e);
			}
		}
		String value = reader.getValue();
		if (null != value && !"".equals(value)) {
			try {
				Field field = obj.getClass().getDeclaredField(mapper.realMember(obj.getClass(), textName));
				if (!field.isAccessible()) {
					field.setAccessible(true);
				}
				field.set(obj, value);
			} catch (Exception e) {
				e.printStackTrace();
				throw new ObjectAccessException("Cannot construct " + obj.getClass(), e);
			}
		}
		return obj;
	}

	public boolean canConvert(Class clazz) {
		/*Iterator it = convertClass.iterator();
		while (it.hasNext()) {
			Class c = (Class) it.next();
			if (c.equals(clazz)) {
				return true;
			}
		}return false;*/
		return convertClass.contains(clazz);
	}

	public Iterator getConvertClass() {
		return convertClass.iterator();
	}

	public void addConvertClass(Class cc) {
		this.convertClass.add(cc);
	}

	public String getTextName() {
		return textName;
	}
	public void setTextName(String textName) {
		this.textName = textName;
	}
}

 

 

 

使用方法:

XStream stream = new XStream();
stream.alias("h-cell", HCell.class);
stream.aliasAttribute(HCell.class, "width", "alias"); 
AttributeValueConveter converter = new AttributeValueConveter(stream.getMapper());  
converter.addConvertClass(PhoneNumber.class);  
stream.registerConverter(converter);
stream.registerConverter(new AttributeValueConveter(stream.getMapper()));

 

分享到:
评论

相关推荐

    XStream组件进行Object与XML互转用法

    - 不推荐无特殊情况地配置属性节点的映射,因为这可能使反序列化过程变得复杂。 - 配置别名和字段别名能极大地提高XML的可读性,同时保持与原始对象结构的一致性。 - 在反序列化时,确保XML的结构与序列化时一致,...

    xml解析及使用XStream实现javaBean与xml之间的转换

    XStream不仅支持XML,还内置了对JSON的支持。只需添加额外的转换器,如使用Jettison库,即可实现JavaBean与JSON的转换。 5. **实战演示** 在`xmlAnalysis`文件夹中,可能包含了一个简单的Java程序,演示了如何...

    xstream最新版 - xstream-1.4.18.jar

    xstream最新版jar包

    com.thoughtworks.xstream.XStream操作XML和java对象的一些用法

    它的主要优点是代码简洁,易于理解,同时也支持自定义转换规则,使得XML的结构可以更贴近业务需求。 1. **Java对象转换成XML字符串** XStream的核心在于`toXML()`方法,它可以将任何Java对象转换成XML字符串。例如...

    Oracle Database XStream Guide 11g Release 2 (11.2)-302

    * XStream 的事件处理引擎支持多种编程语言,包括 Java、C++ 和 Python 等。 三、数据传输 * Oracle Database XStream 支持多种数据传输协议,包括 TCP/IP、HTTP 和 HTTPS 等。 * XStream 可以将捕获的数据传输到...

    xstream-1.4.2

    它的出现为开发者提供了便利,使得处理XML数据如同操作Java对象一般简单。在XStream 1.4.2这个版本中,它继续优化了性能,增强了安全性,并且保持了对多种Java框架的良好兼容性。本文将深入探讨XStream 1.4.2中的...

    xStream完整Jar包

    5. **XML 格式**:XStream 输出的 XML 结构清晰,易于阅读,同时也支持 JSON 格式。 **使用 XStream 的步骤** 1. 引入 XStream 库:将“xStream完整jar包”中的 JAR 文件添加到项目类路径中。 2. 创建 XStream ...

    xstream-1.4-API

    - **自定义对象**: 只需定义好 Java 类,XStream 就能自动处理其属性和嵌套对象。 ### 4. **映射 XML 和 Java 类** - **字段映射(Field Aliases)**: 使用 `aliasField()` 方法可以为类的字段设置别名。 - **类...

    XStream解析各种数据格式

    XStream 是一个简单高效的 Java 库,用于将 Java 对象序列化为 XML 文档,同时也支持从 XML 文档反序列化为 Java 对象。它提供了简洁的 API 来简化序列化过程,使开发者能够轻松地处理对象与 XML 之间的转换。 ####...

    xstream-1.4.2.jar

    5. **支持泛型**:XStream能够处理带有泛型的类,保留类型信息。 6. **流式处理**:XStream可以将XML解析为事件流,这对于处理大型XML文件时可以节省内存。 7. **支持JSON**:除了XML,XStream还支持JSON格式的...

    XStream XML与Json转换

    在缺省情况下,XStream不需要配置映射关系,对象和字段将映射为同名XML元素。但是当对象和字段名与XML中的元素名不同时,XStream支持指定别名。XStream支持以方法调用的方式,或是Java 标注的方式指定别名。 ...

    xstream解析依赖包

    例如,XStream提供了内置的转换器处理基本类型、集合、日期等,同时也支持用户自定义转换器以处理特定的复杂对象。 XStream-1.3.1版本是一个相对早期的稳定版本,它包含了对JDK1.4及以上的支持,同时提供了许多增强...

    XStream+1.3(xstream).CHM

    为了更好地理解和利用XStream,开发者应深入研究其API文档,学习如何配置转换器,处理嵌套的对象关系,以及如何处理XML的命名空间和属性。 总的来说,XStream 1.3是一个功能强大且易于使用的XML序列化库,它简化了...

    xstream-1.3.1.zip

    XStream还支持处理集合和数组,可以将它们转换为XML列表,也可以从XML列表中恢复。此外,对于嵌套的对象和复杂的继承结构,XStream也能处理得游刃有余。 总结起来,XStream作为一个强大的Java库,极大地简化了...

    xstream-1.4.15.jar

    Xstream上次对CVE-2020-26217处理并不彻底,虽然通过黑名单方法阻止了远程代码执行,但是仍然可以采用类似思路实现文件删除与服务器请求伪造。 影响版本 Xstream 修复版本 Xstream &gt; = 1.4.15 风险等级 严重

    xstream-1.4.8最齐全的jar包

    4. **域级注解**:XStream提供了`@XStreamAlias`、`@XStreamAsAttribute`等注解,允许开发者在类或字段级别控制XML的生成方式,比如改变元素名或将其作为属性处理。 5. **集合处理**:XStream能够自动处理Java集合...

    Xstream 相关jar包

    3. **自动类型转换**:Xstream支持自动类型转换,这意味着即使XML中的值是字符串类型,也能正确地转换回对应的Java类型,如日期、数字等。 4. **自定义转换器**:对于标准转换器无法处理的复杂对象或自定义类型,...

    Xstream.Core 源代码 C#

    4. 循环引用处理:Xstream.Core通过跟踪已序列化的对象和其对应的ID,避免了在序列化过程中出现无限递归的问题。 5. 自定义转换器:源代码中,`ConverterRegistry`管理着所有自定义的类型转换器,允许开发者为特定...

    XStream jar包及源码

    此外,XStream还支持XML的命名空间(namespace),使得在处理具有复杂XML结构的应用中更加灵活。通过配置,你可以控制XML元素的命名,甚至改变默认的标签名称和属性名。 总之,XStream是一个强大且灵活的工具,能够...

    XStream使用文档 wd.docx

    6. **详尽的错误日志**:当序列化或反序列化出现问题时,XStream 提供了详细的错误信息。 7. **多格式支持**:除了XML,XStream 还可以将 Java 对象转换为 JSON 格式。 **XStream 的优势** 与传统的序列化框架相比...

Global site tag (gtag.js) - Google Analytics