繁体   English   中英

休眠:使用executeUpdate()批量删除不会清除一级缓存

[英]Hibernate: Batch delete with executeUpdate() does not clear first level cache

我的意图是使用executeUpdate()执行单个DELETE,而不是遍历session.delete() ,这令人难以置信。

我面临的问题是通过执行executeUpdate() ,它没有从一级缓存中逐出现有已删除的实体。

这是一个非常简单的示例:

final Session session = sessionFactory.getCurrentSession();

System.out.println("---------------------------------");
System.out.println("What's currently in 1st level cache");
System.out.println("---------------------------------");

for (Object o : session.getStatistics().getEntityKeys()) {
    EntityKey entityKey = (EntityKey) o;
    System.out.println(entityKey.getEntityName() + " -> " + entityKey.getIdentifier());
}

System.out.println("---------------------------------");
System.out.println("Ensure there are 3 project users");
System.out.println("---------------------------------");

checkArgument(session.createQuery("from ProjectUser").list().size() == 3);

System.out.println("---------------------------------");
System.out.println("Deleting all 3 project users");
System.out.println("---------------------------------");

checkArgument(session.createQuery("delete from ProjectUser").executeUpdate() == 3);

session.flush();

System.out.println("---------------------------------");
System.out.println("Latest state in 1st level cache");
System.out.println("---------------------------------");

for (Object o : session.getStatistics().getEntityKeys()) {
    EntityKey entityKey = (EntityKey) o;
    System.out.println(entityKey.getEntityName() + " -> " + entityKey.getIdentifier());
}

这是我看到的打印输出:-

---------------------------------
What's currently in 1st level cache
---------------------------------
testHibernate.domain.ProjectUser -> 2
testHibernate.domain.ProjectUser -> 1
testHibernate.domain.User -> 1
testHibernate.domain.ProjectUser -> 3
testHibernate.domain.User -> 3
testHibernate.domain.Project -> 1
testHibernate.domain.User -> 2
---------------------------------
Ensure there are 3 project users
---------------------------------
Hibernate: 
    select
        projectuse0_.projectUserId as projectU1_1_,
        projectuse0_.datetime as datetime2_1_,
        projectuse0_.projectId as projectI3_1_,
        projectuse0_.userId as userId4_1_ 
    from
        projectUser projectuse0_
[TRACE] [BasicExtractor] [extract:74] - Found [1] as column [projectU1_1_]
[TRACE] [BasicExtractor] [extract:74] - Found [2] as column [projectU1_1_]
[TRACE] [BasicExtractor] [extract:74] - Found [3] as column [projectU1_1_]
---------------------------------
Deleting all 3 project users
---------------------------------
Hibernate: 
    delete 
    from
        projectUser
---------------------------------
Latest state in 1st level cache
---------------------------------
testHibernate.domain.ProjectUser -> 2
testHibernate.domain.ProjectUser -> 1
testHibernate.domain.User -> 1
testHibernate.domain.ProjectUser -> 3
testHibernate.domain.User -> 3
testHibernate.domain.Project -> 1
testHibernate.domain.User -> 2

如您所见, testHibernate.domain.ProjectUser实体仍在第一级缓存中徘徊,即使它们已被删除。

我的问题是如何使用executeUpdate()方法从一级缓存中删除已删除的实体? 是否有任何官方文档解释此问题?

谢谢。

这是我的解决方案,但确实很麻烦。

BaseEntity只是我们所有实体的基类。

@PersistenceContext
transient EntityManager entityManager;

@SuppressWarnings("unchecked")
public void evict(Class<? extends BaseEntity> c)
{
    HibernateEntityManager hem = entityManager.unwrap(HibernateEntityManager.class);

    Session session = hem.getSession();

    Query query = session.createQuery("select o from " + c.getSimpleName() + " o where isObsolete = false AND isDeleted = false");

    List<? extends BaseEntity> entities = (List<? extends BaseEntity>) query.list();

    for (BaseEntity e : entities)
    {
        session.evict(e);
    }
}

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM