JAXB,Hibernate,Lazy loading

时间:2013-02-22 15:52:47

标签: hibernate jaxb resteasy

我在下面粘贴了一个简单的Hibernate POJO(为简洁起见,删除了构造函数和setter)。我的问题出现在“用户”关系上。 Hibernate延迟加载关系很好,但是当我的CRUD webservice调用(也在下面)调整此对象的实例时,它调用关系的“get”方法,因此在Hibernate中抛出“No transaction”异常,因为JAXB没有访问会话或交易中的关系。

POJO:

@Entity
@Table(name = "ldapservers", uniqueConstraints = @UniqueConstraint(columnNames = "hostname"))
@XmlRootElement(name = "ldap-server")
@SuppressWarnings("unused")
public class LdapServer implements Serializable
{
    private int ldapServerId;
    private String hostname;
    private int port;
    private Date createDate;
    private String createUser;
    private Set<User> users = new HashSet<User>(0);
    @Id 
    @GeneratedValue(strategy=GenerationType.IDENTITY)
    @Column(name = "ldapServerID", unique = true, nullable = false)
    @XmlAttribute(name="id")
    public int getLdapServerId()
    {
        return this.ldapServerId;
    }
    @Column(name = "hostname", unique = true, nullable = false)
    @XmlElement
    public String getHostname()
    {
        return this.hostname;
    }
    @Column(name = "port", nullable = false)
    @XmlElement
    public int getPort()
    {
        return this.port;
    }
    @Temporal(TemporalType.TIMESTAMP)
    @Column(name = "createDate", nullable = false, length = 19)
    @XmlAttribute(name="create-date")
    public Date getCreateDate()
    {
        return this.createDate;
    }
    @Column(name = "createUser", nullable = false)
    @XmlAttribute(name="create-user")
    public String getCreateUser()
    {
        return this.createUser;
    }
    @OneToMany(fetch = FetchType.LAZY, mappedBy = "ldapServer")
    public Set<User> getUsers()
    {
        return this.users;
    }
}

WebService Call:

    @GET
@Path("/fetch/{id}")
@Produces("application/xml")
public LdapServer getLdapServer(@PathParam("id") int ldapServerID)
{
    logger.debug("Fetching LdapServer ID "+ldapServerID);
    LdapServer ls = this.home.findById(ldapServerID);

    if (ls!=null)
    {
        logger.debug("Found LdapServer ID "+ldapServerID);
    }
    else
    {
        logger.debug("LdapServer ID "+ldapServerID+" not found.");
    }

    return ls;
}

我没有包含DAO / EJB代码,因为错误发生在Resteasy内部和此调用之外,表明在编组期间出现问题。

1 个答案:

答案 0 :(得分:0)

我也有同样的问题。我假设您在resteasy方法开始时打开了一个hibernate会话和事务,并且在方法结束时关闭它。问题是延迟加载会将代理对象返回给用户。您可以逐步调试它并运行ls.getUsers()。类型是代理。一旦您访问该代理中的某些内容,它将仅获取实际对象。在您的情况下,这会在编组期间发生,但到那时,您的会话/事务已关闭,因此您的错误会出现。如果你在resteasy方法中访问类似ls.getUsers.size()的东西,代理现在将是你期望的实际对象,并且编组将起作用,但是必须这样做似乎很麻烦。回到我遇到问题时,我只是决定急切地避免这种混乱。但是,现在有一个修复程序,您可以在此处查看https://hibernate.onjira.com/browse/HHH-7457