0

在问这个问题之前,这是我的问题的背景:

我有一个名为“Category”的特定实体,它有一个属性 Category(父级)和另一个类型为“relatedEntity”的属性。

这是代码示例:

@Entity
public class Category {

    ...

    @ManyToOne(fetch=FetchType.LAZY)
    @JoinColumn(name="entityID")
    private RelatedEntity entity;

    @ManyToOne(fetch=FetchType.LAZY)
    @JoinColumn(name="parentID")
    private Category parent;

    ...
}

因此,如您所见,我已经为这些属性声明了 ManyToOne 关系,并且还声明了延迟加载获取策略。

现在,我在 DAO 对象中使用此方法来获取类别列表:

public List<Category> getAll() {
    Criteria crit = sessionFactory.getCurrentSession().createCriteria(Category.class);
    return (crit.list());
}

我得到了一个调用 DAO 方法的特定服务对象:

@Transactional(readOnly = true)
public List<Category> getAllCategories() {
    return (categoryDAO.getAll());
}

目前没有什么特别的……现在,我的单元测试来了:

@Test
public void testCategories() {

    List<Category> cat = service.getAllCategories();
    assertNotNull(cat);
    assertFalse(cat.isEmpty());

    for (Category c : cat) {

        try {
            assertNotNull(c.getEntity().getName());
            fail("Expected lazy init. on Category.Entity");
        }
        catch(LazyInitializationException ex) {
             //Exception is caught
        }

        try {
            assertNotNull(c.getParent().getName());
            //No exception
            fail("Expected lazy init. on Category.Parent");
        }
        catch(LazyInitializationException ex) {}
    }
}

运行测试后,在第一个 try/catch 块中捕获了一个异常,并触发了第二个失败,表示该类别的父属性上没有 LazyInitException。

所以我可以打电话

c.getParent().getParent().getName(); 

或者

c.getParent().getParent().getParent().getName() 

没有任何异常被提出。

我在我的配置文件中设置了'show_sql=true',在调用父属性的getter后,我在日志中看不到任何补充请求,所以没有新请求,但加载了父属性。

有人能解释一下为什么父属性不懒惰吗?

编辑:

我已经更新了我的测试用例。

@Test(expected=LazyInitializationException.class)   
public void lazyCategoriesParentsList() {
    List<Category> cat = service.getAllCategories();

    assertNotNull(cat);
    assertFalse(cat.isEmpty());

    for (Category c : cat) {            
        assertNotNull(c.getParent().getName());     
    }       
}

现在测试通过了......它是否与我之前的测试用例中实体属性的访问(通过c.getEntity())有关?

4

1 回答 1

3

您可以尝试只获取一个类别而不是全部加载吗?

事实上,当你加载一个Category时,如果 hibernate 可以在同一个Session中找到它的级,它会初始化它,即使级被标记为惰性。当您查询所有类别时,它将找到Session中的所有父级并初始化它们。

于 2012-11-14T10:30:26.963 回答