【问题标题】:Hibernate interceptor or listener with Spring Boot and Spring Data JPA使用 Spring Boot 和 Spring Data JPA 的 Hibernate 拦截器或侦听器
【发布时间】:2017-02-01 11:38:40
【问题描述】:

我想在保存对象的一组子对象之前运行一些检查(级联 = 全部)。

我正在使用 Spring Boot 和 Spring Data JPA,并且想知道哪种方法最好:Hibernate 侦听器或拦截器。每个的优点/缺点是什么?您是否碰巧有一个您认为最好的方法的示例?

我在 XML 中配置之前使用过 Hibernate 侦听器,如下所示:

    <property name="eventListeners">
        <map>
            <entry key="post-update">
                <list>
                    <ref bean="myListener" />
                </list>
            </entry>
        </map>
    </property>

关于会话工厂(旧项目)。但是现在我的大部分配置都在注释中(因为 Spring Boot),我想让配置尽可能简单和轻便,所以也许拦截器会是一个更好的解决方案。

谢谢。

【问题讨论】:

    标签: spring hibernate spring-boot spring-data spring-data-jpa


    【解决方案1】:

    我为自己做了很多环顾四周,并认为我会分享我的工作(我在底部添加了有用的(非内联)链接)。

    拦截器

    要使用拦截器,您可以扩展org.hibernate.EmptyInterceptor 类并覆盖您要拦截的方法。 您可能需要onSave(...)

    package foo.bar;
    
    import org.hibernate.EmptyInterceptor;
    import org.hibernate.type.Type;
    import java.io.Serializable;
    
    public class MyInterceptor extends EmptyInterceptor {
        @Override
        public boolean onSave(Object entity, Serializable id, Object[] state, String[] propertyNames, Type[] types) {
            // do your checks here
            return false;
        }
    }
    

    您必须使用 Spring/Hibernate register your interceptor。 您可以在application.properties or application.yml 中执行此操作。

    spring:
      jpa:
        properties:
          hibernate.ejb.interceptor: foo.bar.MyInterceptor
    

    拦截器的好处是它(可能)更少的代码和相对简单的配置。 缺点是您的整个应用程序只能使用一个,并且使用该 API 可能会令人困惑。

    事件监听器

    对于事件,您实现 Hibernate 的 org.hibernate.event.spi.*Listener 接口之一。 您可能需要 org.hibernate.event.spi.PreInsertEventListener 在您的情况下。

    您必须在EventListenerRegistry 中注册您的活动。 为此,您可以将您的类设为@Component@AutowireEntityManagerFactory,并创建一个@PostConstruct 方法来注册您的类。

    package foo.bar;
    
    import org.hibernate.event.service.spi.EventListenerRegistry;
    import org.hibernate.event.spi.EventType;
    import org.hibernate.event.spi.PreInsertEvent;
    import org.hibernate.event.spi.PreInsertEventListener;
    import org.hibernate.internal.SessionFactoryImpl;
    import org.springframework.beans.factory.annotation.Autowired;
    import org.springframework.stereotype.Component;
    import javax.annotation.PostConstruct;
    import javax.persistence.EntityManagerFactory;
    
    @Component
    public class MyEventListener implements PreInsertEventListener {
        @Autowired
        private EntityManagerFactory entityManagerFactory;
    
        @PostConstruct
        private void init() {
            SessionFactoryImpl sessionFactory = entityManagerFactory.unwrap(SessionFactoryImpl.class);
            EventListenerRegistry registry = sessionFactory.getServiceRegistry().getService(EventListenerRegistry.class);
            registry.getEventListenerGroup(EventType.PRE_INSERT).appendListener(this);
        }
    
        @Override
        public boolean onPreInsert(PreInsertEvent preInsertEvent) {
            // do your checks here
            return false;
        }
    }
    

    侦听器的好处是您可以拥有任意数量的侦听器,API 比拦截器更好,并且代码和配置都在一个地方。 缺点是配置更长,涉及更多。


    【讨论】:

    • Hibernate 支持会话范围的拦截器以及应用范围的拦截器,您需要使用此属性而不是“hibernate.ejb.interceptor.session_scoped”
    • 太棒了,我看了很多,得到了和你一样好的指导。
    • 精彩的解释兄弟!
    【解决方案2】:

    你好

    首先您可以查看:https://www.baeldung.com/database-auditing-jpa,其中详细解释了每个选项。

    我个人会推荐Hibernate Interceptor,易于使用和理解。根据项目的复杂程度,在大多数情况下都可以。

    为了在您的应用程序中配置它,您只需添加:spring.jpa.properties.hibernate.ejb.interceptor = path.to.interceptor(在 application.properties 中)。拦截器本身应该是@Component

    只要拦截器实际上不使用任何 bean。否则会有点复杂,但我很乐意提供解决方案。

    不要忘记添加 application-test.properties,一个 EmptyInterceptor 在测试中不使用日志系统(或任何你想使用它的东西)(这不会很有帮助)。

    希望这对你有用。

    最后一点:始终更新您的 Spring / Hibernate 版本(尽可能使用最新版本),您会发现大多数代码将变得多余,因为新版本会尽可能减少配置可能。

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 2019-07-07
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2018-08-09
      • 2014-10-06
      • 1970-01-01
      • 1970-01-01
      相关资源
      最近更新 更多