我将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>
不要忘记重启服务器