java - 使用要创建的类参数化 JAXB 工厂方法

标签 java jaxb

根据文档,JAXB 工厂方法没有参数。是否有 JAXB 实现允许我创建一个工厂方法,该方法接收我需要创建的对象的类作为参数? 碰巧我所有的 JAXB 对象都遵循相同的创建模式(一种特定的字节码检测),因此我想将其封装在一个工厂方法中,将要创建的 JAXB 对象的类作为参数,从而避免这种方式为基本上做完全相同事情的每个 JAXB 类创建不同的工厂方法。

我发现有人在 OTN 论坛上问同样的问题:https://forums.oracle.com/forums/thread.jspa?messageID=9969927#9969927 ,但尚未提出真正的答案。

感谢您的帮助

最佳答案

目前使用标准 JAXB API 无法做到这一点。我已输入以下增强请求以将此行为添加到 EclipseLink JAXB (MOXy) :

MOXy 特定解决方案

您可以利用 EclipseLink JAXB (MOXy) 中的 @XmlCustomizer 扩展自定义对象的实例化方式。此机制用于调整 MOXy 的底层元数据。

CommonFactory

import java.util.Date;

public class CommonFactory {

    public static Object create(Class<?> clazz) {
        if(Foo.class == clazz) {
            return new Foo(new Date());
        } else if(Bar.class == clazz) {
            return new Bar(new Date());
        }
        return null;
    }

}

Foo.class

Foo 类被正常注释,除了我们将使用 @XmlCustomizer 注释来指定我们将要使用的 DescriptorCustomizer调整 MOXy 的元数据。

import java.util.Date;
import javax.xml.bind.annotation.*;
import org.eclipse.persistence.oxm.annotations.XmlCustomizer;

@XmlRootElement
@XmlType(factoryClass=CommonFactory.class, factoryMethod="create")
@XmlAccessorType(XmlAccessType.FIELD)
@XmlCustomizer(FactoryCustomizer.class)
public class Foo {

    private Date creationDate;
    private Bar bar;

    // Non-default constructor
    public Foo(Date creationDate) {
        this.creationDate = creationDate;
    }

}

酒吧

我们将再次使用 @XmlCustomizer 注释来引用我们在 Foo 类中所做的相同的 DescriptorCustomizer

import java.util.Date;
import javax.xml.bind.annotation.*;
import org.eclipse.persistence.oxm.annotations.XmlCustomizer;

@XmlType(factoryClass=CommonFactory.class, factoryMethod="create")
@XmlAccessorType(XmlAccessType.FIELD)
@XmlCustomizer(FactoryCustomizer.class)
public class Bar {

    private Date creationDate;

    // Non-default constructor
    public Bar(Date creationDate) {
        this.creationDate = creationDate;
    }

}

工厂定制器

MOXy 有一个InstantiationPolicy 的概念来构建新对象。在此示例中,我们将交换我们自己的实例 InstantiationPolicy,它可以使用参数化工厂方法:

import org.eclipse.persistence.config.DescriptorCustomizer;
import org.eclipse.persistence.descriptors.ClassDescriptor;
import org.eclipse.persistence.exceptions.DescriptorException;
import org.eclipse.persistence.internal.descriptors.InstantiationPolicy;
import org.eclipse.persistence.internal.security.PrivilegedAccessHelper;
import org.eclipse.persistence.internal.sessions.AbstractSession;

public class FactoryCustomizer implements DescriptorCustomizer{

    @Override
    public void customize(ClassDescriptor descriptor) throws Exception {
        descriptor.setInstantiationPolicy(new MyInstantiationPolicy(descriptor));
    }

    private static class MyInstantiationPolicy extends InstantiationPolicy {

        public MyInstantiationPolicy(ClassDescriptor descriptor) {
            InstantiationPolicy defaultInstantiationPolicy = descriptor.getInstantiationPolicy();
            this.factoryClassName = defaultInstantiationPolicy.getFactoryClassName();
            this.factoryClass = defaultInstantiationPolicy.getFactoryClass();
            this.methodName = defaultInstantiationPolicy.getMethodName();
        }

        @Override
        public void initialize(AbstractSession session) throws DescriptorException {
            super.initialize(session);
        }

        @Override
        protected void initializeMethod() throws DescriptorException {
            Class<?>[] methodParameterTypes = new Class[] {Class.class};
            try {
                this.method = PrivilegedAccessHelper.getMethod(factoryClass, methodName, methodParameterTypes, true);
            } catch (NoSuchMethodException e) {
                throw new RuntimeException(e);
            }
        }

        @Override
        public Object buildNewInstance() throws DescriptorException {
            Object[] parameters = new Object[] {this.descriptor.getJavaClass()};
            try {
                return PrivilegedAccessHelper.invokeMethod(method, factory, parameters);
            } catch (Exception e) {
                throw new RuntimeException(e);
            }
        }

    }

}

演示

import java.io.StringReader;
import javax.xml.bind.*;

public class Demo {

    public static void main(String[] args) throws Exception {
        JAXBContext jc = JAXBContext.newInstance(Foo.class);

        Unmarshaller unmarshaller = jc.createUnmarshaller();
        Foo foo = (Foo) unmarshaller.unmarshal(new StringReader("<foo><bar/></foo>"));

        Marshaller marshaller = jc.createMarshaller();
        marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);
        marshaller.marshal(foo, System.out);
    }

}

输出

<?xml version="1.0" encoding="UTF-8"?>
<foo>
   <creationDate>2011-11-08T12:35:43.198</creationDate>
   <bar>
      <creationDate>2011-11-08T12:35:43.198</creationDate>
   </bar>
</foo>

了解更多信息

关于java - 使用要创建的类参数化 JAXB 工厂方法,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/8026502/

相关文章:

java - 如何解码与 Java 的 JAXB 库的非层次关系?

java - JAXB:按特定顺序排列的多个同类标签

java - 重新保存对象而不是在列表中创建新对象

java - 列计数与 java 和 mysql 数据库中第 1 行的值计数不匹配

java - C# 实现 - "Print all combinations of balanced parentheses"

java - 创建抽象通用 jaxb 类

java - 将 XML 转换为 Java 对象时获取不正确的值

java - 如何在 iText 7 中用字符填充页面宽度

java - 尝试在空对象引用上调用虚拟方法 'boolean java.util.ArrayList.add(java.lang.Object)'

java - 支持 SAX 2.0 compilant 解析器