Jackson XML注释:使用属性

时间:2017-04-26 18:39:52

标签: java xml jackson pojo

我使用Jackson XML注释将XML文档从外部API转换为POJO。 XML中的一个元素给我带来了一些麻烦。大多数元素没有属性,只有文本值,如:

<title>Title Here</title>

我在使用一个元素时遇到了一些麻烦,它有一个属性,如下所示:

<urgency id="UrgCaution">Caution</urgency>

我只想提取文字值&#34;警告&#34;并将其存储在一个字符串中。我最初在我的Java类中尝试过这种方式:

public class Item {
    @JacksonXmlProperty(localName = "urgency")
    private String urgency;
}

但是这会导致以下错误:

Caused by: com.fasterxml.jackson.databind.JsonMappingException: Can not construct 
instance of Item: no String-argument constructor/factory method to deserialize from 
String value ('Security')
at [Source: java.io.ByteArrayInputStream@328ff654; line: 40, column: 21] 

&#34;安全&#34;是一个名为&#34;标记&#34;的xml属性的文本。稍后会出现在文件中。

我尝试进行以下更改,这消除了异常,但导致我在POJO中获得紧急程度的空值。

public class Item {
    @JacksonXmlProperty(localName = "urgency")
    @JacksonXmlText
    private String urgency;
}

我觉得我在这里遗漏了一些明显的东西。使用Jackson将这个元素的文本转换为POJO上的String字段的最佳方法是什么?

这是我正在使用的完整课程和XML文档。

XML:

<rss version="2.0">
    <channel>
        <title>Title</title>
        <description>Description</description>
        <copyright>Copyright info</copyright>
        <ttl>5</ttl>
        <pubDate>Thu, 27 Apr 2017 17:21:40 GMT</pubDate>
        <item>
            <title>Title of event in U.S.</title>
            <description>Description here</description>
            <fulldescription>Full description here</fulldescription>
            <link>http://www.website.com</link>
            <pubDate>Thu, 27 Apr 2017 17:13:48 GMT</pubDate>
            <locations>
                <location>
                    <name>New York, NY</name>
                    <latitude>40.714502</latitude>
                    <longitude>-74.006998</longitude>
                    <code>NYC</code>
                    <city>New York</city>
                    <state>New York</state>
                    <country>United States</country>
                    <region>North America</region>
                </location>
            </locations>
            <source>AP</source>
            <urgency id="UrgAttention">Attention</urgency>
            <categories>
                <category id="CatTransp">Transportation</category>
            </categories>
            <destinations>
                <destination code="NYC" id="US" lat="40.714502" longitude="-74.006998">New York, New York</destination>
            </destinations>
            <designations />
            <related />
            <tags>Railway disruptions</tags>
            <guid>3d64388fc639475dc9aeaabf81ee87bd</guid>
        </item>
    </channel>
</rss>

现在上课了。为简洁起见,省略了吸气剂和固定剂。还有位置,目的地和类别类别,但除非有必要,否则我不会包含这些类别。

根类:

@JacksonXmlRootElement(localName = "rss")
public class FeedRoot {
    @JacksonXmlProperty(localName = "channel")
    private FeedChannel channel;
}

频道类:

public class FeedChannel {
    @JacksonXmlElementWrapper(localName = "item", useWrapping = false)
    @JacksonXmlProperty(localName = "item")
    private List<Item> items;
}

Item Class(Json属性用于将结果编组为JSON,但删除它们不会改变结果):

public class Item {
    @JacksonXmlProperty(localName = "title")
    private String title;

    @JacksonXmlProperty(localName = "description")
    private String description;

    @JacksonXmlProperty(localName = "fulldescription")
    @JsonProperty("full_description")
    private String fullDescription;

    @JacksonXmlProperty(localName = "link")
    private String link;

    @JacksonXmlProperty(localName = "pubDate")
    @JsonProperty("publish_date")
    private String pubDate;

    @JacksonXmlElementWrapper(localName = "locations")
    @JacksonXmlProperty(localName = "location")
    private List<Location> locations;

    @JacksonXmlProperty(localName = "source")
    private String source;

    @JacksonXmlProperty(localName = "urgency")
    private String urgency;

    @JacksonXmlElementWrapper(localName = "categories")
    @JacksonXmlProperty(localName = "category")
    private List<Category> categories;

    @JacksonXmlElementWrapper(localName = "destinations")
    @JacksonXmlProperty(localName = "destination")
    private List<Destination> destinations;

    @JacksonXmlProperty(localName = "related")
    private String related;

    @JacksonXmlProperty(localName = "tags")
    private String tags;

    @JacksonXmlProperty(localName = "guid")
    private String guid;

    //getters and setters below here, no annotations
}

尝试解析时出错(省略了名称空间):

"Could not read document: Can not construct instance of 
com.namespacestuff.Item: no String-argument constructor/factory 
method to deserialize from String value ('Railway disruptions')\n at 
[Source: java.io.ByteArrayInputStream@45563ea1; line: 42, column: 32] 
(through reference chain: com.namespacestuff.FeedRoot[\"channel\"]-
>com.namespacestuff.FeedChannel[\"item\"]->java.util.ArrayList[5]); 
nested exception is 
com.fasterxml.jackson.databind.JsonMappingException: Can not 
construct instance of com.namespacestuff.Item: no String-argument 
constructor/factory method to deserialize from String value ('Railway 
disruptions')\n at [Source: java.io.ByteArrayInputStream@45563ea1; 
line: 42, column: 32] (through reference chain: 
com.namespacestuff.FeedRoot[\"channel\"]-
>com.namespacestuff.FeedChannel[\"item\"]->java.util.ArrayList[5])"

2 个答案:

答案 0 :(得分:3)

<强>更新

由于OP在我的回答后更新,我加上一些考虑。

即使提供了代码,如果我尝试反序列化 Item 元素,一切都很好。

当涉及所有 rss 元素时,就会出现问题。

查看XML,问题在 FeedChannel 定义上显示。

public class FeedChannel {
    @JacksonXmlElementWrapper(localName = "item", useWrapper = false)
    @JacksonXmlProperty(localName = "item")
    private List<Item> items;
}

我尝试将此 useWrapper localName 更改为 items ,然后我相应地更改了XML。

<items><item>...</item></items>

一切顺利。

正如我在下面写的那样, urgency 属性不需要 @JacksonXmlText 注释。

它似乎是 Jackson 上的一个错误,因为 useWrapper 不仅改变了当前列表的行为。

紧急属性 id 将属性转换为数组。 如果删除该属性,一切都很好。

到目前为止,我认为这是杰克逊图书馆的一个问题,你应该向他们报告此案。

原始回答

您错过了发布所需的所有代码,因为您的错误并非我认为的地方。

我做了一个有效的例子:

这是我的POJO:

@JacksonXmlRootElement(localName = "item")
public class Item {
    @JacksonXmlProperty(localName = "field1")
    private String field1;

    @JacksonXmlProperty(localName = "field2")
    private String field2;

    @JacksonXmlProperty(localName = "urgency")
    private String urgency;

    @JacksonXmlProperty(localName = "tags")
    private String tags;

    public String getField1() {
        return field1;
    }

    public void setField1(String field1) {
        this.field1 = field1;
    }

    public String getField2() {
        return field2;
    }

    public void setField2(String field2) {
        this.field2 = field2;
    }

    public String getUrgency() {
        return urgency;
    }

    public void setUrgency(String urgency) {
        this.urgency = urgency;
    }

    public String getTags() {
        return tags;
    }

    public void setTags(String tags) {
        this.tags = tags;
    }
}

以下是测试:

public class ItemTest {

    @Test
    public void readItemTest() {

        final String itemXml = "<item>\n" +
                "    <field1>Field1 Text</field1>\n" +
                "    <field2>Field2 Text</field2>\n" +
                "    <urgency id=\"UrgCaution\">Caution</urgency>\n" +
                "    <tags>tag</tags>\n" +
                "</item>";

        XmlMapper xm = new XmlMapper();

        try {
            Item myItem = xm.readValue(itemXml, Item.class);

            assert("Caution".equals(myItem.getUrgency()));
        } catch (IOException e) {
            e.printStackTrace();
            assert(false);
        }

    }
}

没有错误,没有问题,只是按预期工作。

Java 7

我的依赖项:

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

    <dependency>
        <groupId>com.fasterxml.jackson.core</groupId>
        <artifactId>jackson-annotations</artifactId>
        <version>2.8.0</version>
    </dependency>

    <dependency>
        <groupId>com.fasterxml.jackson.dataformat</groupId>
        <artifactId>jackson-dataformat-xml</artifactId>
        <version>2.8.0</version>
    </dependency>

答案 1 :(得分:0)

要拥有这个

<urgency id="UrgCaution">Caution</urgency>

您必须创建如下的POJO:

@JacksonXmlRootElement(localName = "urgency")
public class Urgency {
    @JacksonXmlProperty(isAttribute=true)
    String id;
    @JacksonXmlText
    String text;
}