在JAX-RS中序列化java.util.list

时间:2012-11-29 10:07:36

标签: java web-services jpa jaxb jax-rs

我有课本:

@Entity
@Table(name = "books")
@XmlRootElement
@XmlAccessorType(XmlAccessType.FIELD)
@XmlType

public class Book {
    @Id
    @XmlAttribute
    private String isbn;

    private String title;
    private String authors;
    private int year;

    @OneToMany(mappedBy="reservedBook")
    private List<Reservation> bookReservations;

    //Getters, setters, addReservation, remove Reservation
        ......................
}

然后,我有课程预约

@Entity
@Table(name = "reservations")
@XmlRootElement

public class Reservation {

    private String username;
    private String isbn;
    @Temporal(TemporalType.DATE)
    private Date date;

    @ManyToOne
    @JoinColumn(name = "isbn")
    private Book reservedBook;

    @ManyToOne
    @JoinColumn(name = "username")
    private User userWhoReserved;

        //Getters and setters

        ...........
 }

在资源类中,我尝试获取这样的特定书:

@GET
@Path("/{isbn}")
@Produces(MediaType.TEXT_XML)
public Book getBookByIsbn(@PathParam("isbn") String isbn) {

    Book book = entityManager.find(Book.class, isbn);

    if (book != null) {
        return book;

    }

}

现在,它没有序列化字段列表bookReservations。我已经尝试了很多在互联网上发现的想法,比如用@XmlElement注释列表的getter或者在其他地方使用其他注释(我现在不记得了),但没有任何效果。

编辑:这是响应的样子:

<book isbn="3333333333">
    <title>Mere Christianity</title>
    <authors>C. S. Lewis</authors>
    <year>2000</year>
</book>

但我也想表明保留意见。

什么似乎是解决方案? 谢谢! 索林

5 个答案:

答案 0 :(得分:3)

我认为问题可能在列表bookReservationsnull

class A{

    private List<String> someList = new ArrayList<String>();

    @XmlElement(name = "some-tag")
    public List<String> getList() {
        return someList;
    }

    public void setSomeList(List<String> someOtherList) {
        this.someList = someOtherList;
    }
}

我每天使用此代码并为我工作(忽略我未提供的其他注释)。

我在这些情况下遇到了问题,但只发现由于某些原因列表为空而没有序列化。尝试通过调试方法检查JAXB是否在setter方法中设置列表。

答案 1 :(得分:1)

我狂野的客人是延迟加载的问题,你可以在OneToMany和ManyToOne注释中添加fetch = FetchType.EAGER。

public class Book {
    @Id
    @XmlAttribute
    private String isbn;

    private String title;
    private String authors;
    private int year;

    @OneToMany(mappedBy="reservedBook",fetch = FetchType.EAGER)
    private List<Reservation> bookReservations;

    //Getters, setters, addReservation, remove Reservation
        ......................
    }

在预订课程中:

 public class Reservation {

    private String username;
    private String isbn;
    @Temporal(TemporalType.DATE)
    private Date date;

    @ManyToOne(fetch = FetchType.EAGER)
    @JoinColumn(name = "isbn")
    private Book reservedBook;

    @ManyToOne
    @JoinColumn(name = "username")
    private User userWhoReserved;

        //Getters and setters

        ...........
 }

答案 2 :(得分:0)

使用@XmlAccessorType.PROPERTY代替FIELD。 JPA impl可能是延迟加载预订列表,此操作将通过访问属性(get方法)触发。

了解更多信息

答案 3 :(得分:0)

解决!但问题不在于序列化,而是由EntityManager调用的方法find()无法提取保留并将它们放入我的book对象这一事实,因此列表保持为空并且未显示相应的元素。因此,似乎应该使用Query:

,而不是使用find()
Query query = entityManager.createQuery("SELECT book FROM Book book WHERE book.isbn = :_isbn ");
query.setParameter("_isbn", isbn);


//if getSingleResult() doesn't find a book, an exception will be thrown
try {
        Book book = (Book)query.getSingleResult();
        return book;
    }
    catch(RuntimeException e)
    {
        //do something
    }

此外,必须从Book类中删除@XmlAccessorType(XmlAccessType.FIELD),并且应使用@XmlElementWrapper(name =“reservations”)和@XmlElement(name =“reservation”)注释列表的getter。

答案 4 :(得分:0)

使用@XmlRootelementnillable="true"进行换行,然后会有一个空的<reservations>

@XmlElement
@XmlElementWrapper(nillable=true)
private List<Reservation> bookReservations;