2011-05-24 110 views
11

我理解所有关于如何使用XMLAdaptersconvert unmappable types,或只是改变某些对象如何序列化/反序列化为XML。如果我使用注释(包级别或其他),一切都很好。问题是我正在尝试更改第三方对象的表示,我无法更改源代码(即为了注入注释)。JAXB XML适配器通过注释工作,但不通过setAdapter

考虑到Marshaller对象有一个manually adding adapters的方法,这应该不成问题。不幸的是,无论我做什么,我都无法以这种方式设置适配器来“踢入”。例如,我有一个类表示XYZ空间中的点(地心坐标)。在我生成的XML中,我希望将其转换为lat/long/altitude(大地坐标)。这里是我的课:

地心

package testJaxb; 

import javax.xml.bind.annotation.XmlAttribute; 
import javax.xml.bind.annotation.XmlRootElement; 

@XmlRootElement 
public class GeocentricCoordinate { 
    // Units are in meters; see http://en.wikipedia.org/wiki/Geocentric_coordinates 
    private double x; 
    private double y; 
    private double z; 

    @XmlAttribute 
    public double getX() { 
     return x; 
    } 
    public void setX(double x) { 
     this.x = x; 
    } 
    @XmlAttribute 
    public double getY() { 
     return y; 
    } 
    public void setY(double y) { 
     this.y = y; 
    } 
    @XmlAttribute 
    public double getZ() { 
     return z; 
    } 
    public void setZ(double z) { 
     this.z = z; 
    } 
} 

大地

package testJaxb; 
import javax.xml.bind.annotation.XmlAttribute; 
import javax.xml.bind.annotation.XmlRootElement; 
/** 
* @see http://en.wikipedia.org/wiki/Geodetic_system 
*/ 
@XmlRootElement 
public class GeodeticCoordinate { 

    private double latitude; 
    private double longitude; 
    // Meters 
    private double altitude; 

    public GeodeticCoordinate() { 
     this(0,0,0); 
    } 

    public GeodeticCoordinate(double latitude, double longitude, double altitude) { 
     super(); 
     this.latitude = latitude; 
     this.longitude = longitude; 
     this.altitude = altitude; 
    } 

    @XmlAttribute 
    public double getLatitude() { 
     return latitude; 
    } 
    public void setLatitude(double latitude) { 
     this.latitude = latitude; 
    } 

    @XmlAttribute 
    public double getLongitude() { 
     return longitude; 
    } 

    public void setLongitude(double longitude) { 
     this.longitude = longitude; 
    } 

    @XmlAttribute 
    public double getAltitude() { 
     return altitude; 
    } 
    public void setAltitude(double altitude) { 
     this.altitude = altitude; 
    } 



} 

GeocentricToGeodeticLocationAdapter

package testJaxb; 
import javax.xml.bind.JAXBContext; 
import javax.xml.bind.JAXBException; 
import javax.xml.bind.Marshaller; 
import javax.xml.bind.annotation.adapters.XmlAdapter; 


/** 
* One of our systems uses xyz coordinates to represent locations. Consumers of our XML would much 
* prefer lat/lon/altitude. This handles converting between xyz and lat lon alt. 
* 
* @author ndunn 
* 
*/ 
public class GeocentricToGeodeticLocationAdapter extends XmlAdapter<GeodeticCoordinate,GeocentricCoordinate> { 

    @Override 
    public GeodeticCoordinate marshal(GeocentricCoordinate arg0) throws Exception { 
     // TODO: do a real coordinate transformation 
     GeodeticCoordinate coordinate = new GeodeticCoordinate(); 
     coordinate.setLatitude(45); 
     coordinate.setLongitude(45); 
     coordinate.setAltitude(1000); 
     return coordinate; 
    } 

    @Override 
    public GeocentricCoordinate unmarshal(GeodeticCoordinate arg0) throws Exception { 
     // TODO do a real coordinate transformation 
     GeocentricCoordinate gcc = new GeocentricCoordinate(); 
     gcc.setX(100); 
     gcc.setY(200); 
     gcc.setZ(300); 
     return gcc; 
    } 
} 

ObjectWithLocation场

package testJaxb; 
import javax.xml.bind.JAXBContext; 
import javax.xml.bind.JAXBException; 
import javax.xml.bind.Marshaller; 
import javax.xml.bind.annotation.XmlRootElement; 

@XmlRootElement 
public class ObjectWithLocation { 

    private GeocentricCoordinate location = new GeocentricCoordinate(); 

    public GeocentricCoordinate getLocation() { 
     return location; 
    } 

    public void setLocation(GeocentricCoordinate location) { 
     this.location = location; 
    } 


    public static void main(String[] args) { 

     ObjectWithLocation object = new ObjectWithLocation(); 

     try { 
      JAXBContext context = JAXBContext.newInstance(ObjectWithLocation.class, GeodeticCoordinate.class, GeocentricCoordinate.class); 
      Marshaller marshaller = context.createMarshaller(); 

      marshaller.setAdapter(new GeocentricToGeodeticLocationAdapter()); 
      marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true); 

      marshaller.marshal(object, System.out); 

     } 
     catch (JAXBException jaxb) { 
      jaxb.printStackTrace(); 
     } 
    } 
} 

输出:

<?xml version="1.0" encoding="UTF-8" standalone="yes"?> 
<objectWithLocation> 
    <location z="0.0" y="0.0" x="0.0"/> 
</objectWithLocation> 

通过使用注释(在我package-info.java文件):

@javax.xml.bind.annotation.adapters.XmlJavaTypeAdapters 
({ 
@javax.xml.bind.annotation.adapters.XmlJavaTypeAdapter(value=GeocentricToGeodeticLocationAdapter.class,type=GeocentricCoordinate.class), 
}) 

package package testJaxb; 

我碰到下面的(所需)XML

<?xml version="1.0" encoding="UTF-8" standalone="yes"?> 
<objectWithLocation> 
    <location longitude="45.0" latitude="45.0" altitude="1000.0"/> 
</objectWithLocation> 

所以我的问题是双重的。

  1. 为什么适配器在注释时工作,但在通过setAdapter方法显式设置时不工作?
  2. 当我有无法注解的类和无法修改其package-info.java以添加注释的问题时,我该如何解决此问题?

回答

8

MarshallersetAdapter(XmlAdapter)用于在初始化XmlAdapter通过对已经与@XmlJavaTypeAdapter注解的属性。下面的链接是一个答案,我利用这种行为:

如果要映射你可以使用EclipseLink JAXB (MOXy)的XML映射文件的第三方类(我是莫西领先):

+0

好啊。这很烦人。我现在看到,我没有足够好的阅读addAdapter的合同。是否有一个原因不暴露'void addAdapter(类 classToAdapt,XMLAdapter )'方法? – I82Much 2011-05-24 14:49:24

+3

@ I82Much - 可以增强JAXB以提供该行为。目前它不支持这个的原因主要是由于性能。 JAXBContext可以在创建时初始化其元数据,并为每个元帅/解组操作使用相同的元数据。如果允许在Marshaller/Unmarshaller级别引入适配器,则JAXB impl需要考虑这种元数据更改的可能性。 – 2011-05-24 15:35:31

+0

确定非常合理的回应。我只是很烦,因为在XStream中进行这种定制非常容易,而且我发现有更多的障碍需要通过JAXB来完成。我会检查MOXY。谢谢您的帮助。 – I82Much 2011-05-24 16:49:53

2

你总是有0注释

marshaller.setAdapter(...)是在您有非默认构造函数初始化的情况下为您的类型适配器分配自定义初始化实例的方法。否则,如果您的适配器只有一个默认构造函数,那么您不需要明确调用.setAdapter(...)方法。

这里是一个更详细的解释有很大答案: JAXB: Isn't it possible to use an XmlAdapter without @XmlJavaTypeAdapter?

JAXB运行时只能接受无参数的构造适配器。(显然JAXBContext不知道应用的模型)

所以谢天谢地有一个选项:D

你可以告诉你的解组器使用UserAdapter的给定实例,而不是由它自己来设置它。

public class Test { 
public static void main(String... args) { 
    JAXBContext context = JAXBContext.getInstance(Event.class); 
    Unmarshaller unmarshaller = context.createUnmarshaller(); 

     UserContext userContext = null; // fetch it from some where 
     unmarshaller.setAdapter(UserAdapter.class, new UserAdapter(userContext)); 

     Event event = (Event) unmarshaller.unmarshal(..); 
    } 
} 

setAdapter方法上都可用的Marshaller & Unmarshaller的

注: setAdapter上编组/解组并不意味着你没有使用@XmlJavaTypeAdapter