ArrayList 属性的延迟初始化失败(EclipseLink -> 休眠迁移)



我将Java EE war网站从GlassFish 4迁移到WildFly

Glassfish使用EclipseLink, WildFly使用Hibernate。我使用Java服务器面与CDI命名的bean。我的一个bean直接调用dao类方法来检索Category对象。Category类有一些属性,其中之一是List商品。默认情况下,这是惰性初始化的(Merchandise是另一个表)。检索到Category对象,但是当尝试使用List商品(惰性初始化)时,持久性包为空,并抛出以下错误:

javax.servlet.ServletException: failed to lazily initialize a collection of role: cz.pscheidl.velkoobchod.domain.Category.merchandise, could not initialize proxy - no Session
我想我知道这是什么问题了。名为EshopBean的@Named bean在调用该对象时不提供任何事务。这在Glassfish中不应该起作用,但不知何故它做到了。在Category对象中设置了List商品的EAGER初始化后,一切都工作了。在相关的命名查询中,JOIN FETCH也是如此。

EshopBean对象看起来像这样:

@Named
@ViewScoped
public class EshopBean implements Serializable {

    @Inject private CategoryDao categoryDao;
    @Inject private MerchandiseDao merchandiseDao;
    @Inject private ActiveSession activeSession;
    @Inject private OrderDao orderDao;
    @Inject private Logger logger;

    private List<Category> categoryList;
    private List<OrderItem> offeredMerchandise;

    @PostConstruct
    @TransactionAttribute(TransactionAttributeType.REQUIRED)
    public void updateOfferedMerchandise() {
        List<Category> allCateogries = categoryDao.findByOffered(true);
        offeredMerchandise = new ArrayList<>();
        categoryList = new ArrayList<>();
        allCateogries.parallelStream().filter(e -> {
        return !e.getMerchandise().isEmpty();
        }).forEach(e -> {
        categoryList.add(e);
        e.getMerchandise().parallelStream().filter(m -> {
            return m.isOffered();}).forEach(m -> {
            offeredMerchandise.add(createOrderItem(m));
            });
        });
    }
}

类别对象看起来像这样(方法省略):

@Entity
@NamedQueries(
        {
            @NamedQuery(name = "findCategoryByName", query = "SELECT c FROM Category c WHERE c.name = :name"),
            @NamedQuery(name = "findAllCategories", query = "SELECT c FROM Category c"),
            @NamedQuery(name = "categoryHasMerchandise", query = "SELECT COUNT(m) FROM Merchandise m WHERE m.category = :categoryId"),
            @NamedQuery(name = "findCategoryByOffered", query = "SELECT c FROM Category c WHERE c.offered = :offered")
        }
)
public class Category implements Serializable {

    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    private long id;
    @Column(unique = true)
    @Size(min = 1, max = 255)
    private String name;
    @Column(nullable = false)
    private boolean offered;
    @OneToMany(fetch = FetchType.LAZY, cascade = CascadeType.REMOVE)
    @JoinColumn(name = "CATEGORY_ID")
    public List<Merchandise> merchandise;
}

CategoryDaoImpl(省略了一些方法):

@Stateless
public class CategoryDaoImpl implements CategoryDao {
    @PersistenceContext
    private EntityManager entityManager;
    @Override
    @TransactionAttribute(TransactionAttributeType.REQUIRED)
    public List<Category> findByOffered(boolean activity) {
        Query findByOfferedQuery = entityManager.createNamedQuery("findCategoryByOffered", Category.class);
        findByOfferedQuery.setParameter("offered", true);
        return findByOfferedQuery.getResultList();
    }
}

方法调用链如下:EshopBean.updateOfferedMerchandise() -> categorydaoimpp . findbyoffered() ->返回到EshopBean.updateOfferedMerchandise()。我猜两者都被注释为事务性的,findByOffered方法被注释为Required。然而,Hibernate在WildFly说没有会话(我认为它期望相同的事务上下文)。

问题摘要:我想保持商品惰性初始化并纠正会话问题,因此在EshopBean.updateOfferedMerchandise()方法中惰性初始化带有商品的包。我真的需要JOIN FETCH或将Lazy初始化设置为false吗?如何正确设置事务以使此代码开始工作?

当应用程序从Glassfish迁移到Wildfly时,就会出现这个问题。不幸的是,与EclipseLink不同,Wildfly的默认Hibernate JPA实现不允许您在上下文关闭后获取惰性关系。

许多解决方案建议使用即时获取进行重构,但这不是一个好的解决方案,因为即时获取会增加响应,并且通过重构数据库访问将风险引入到工作应用程序中。

宁愿用EclipseLink代替Hibernate作为Wildfly中的JPA持久性提供者。这个实用程序类库将EclipseLink集成到Wildfly中,说明在这里。

你要做的是复制eclipselink-2.6.0.jar(或任何版本你正在使用)到[WILDFLY_HOME]/modules/system/layers/base/org/eclipse/persistence/main,然后编辑module.xml在同一文件夹中包括JAR:

<resources>
   <resource-root path="jipijapa-eclipselink-1.0.1.Final.jar"/>
   <resource-root path="eclipselink-2.6.0.jar">           
      <filter>
         <exclude path="javax/**" />
      </filter>
   </resource-root>
</resources>

不要忘记重启服务器

最新更新