http://blog.csdn.net/shizhan1881/article/details/8478923
最近项目需要,用到了Hibernate的拦截器和监听器,有些小小心得,和大家分享下。
首先说说这两个东西。
拦截器(Intercept):顾名思义,拦截操作,也就是在Hibernate做出动作之前会调用的方法。如果你有需要在Hibernate操作数据库之前想要做的操作,就需要用到这个东西了。
监听器(Listener):监听,就是监视Hibernate的一举一动,如果我们要获取Hibernate各种操作PO的前前后后的信息,那就要用到他了。
这里有的朋友可能就有疑问了,从上面的描述来看,这个监听器似乎能够实现拦截器的功能,因为他也能获取Hibernate操作数据库前的状况。
其实不然,在此我只举出两个很典型的例子,大家就会明白他们俩是不能被互相取代的。
1、监听器只会默默的获取信息,不会阻断Hibernate的工作,而用拦截器时,我们可以根据我们的需求,去终止某个Hibernate的持久化动作。
2、如果我有这样一个需求,在PO保存时,我想按我的需求改变某个属性的值后,再保存入库。分析下知道,这个操作要在Hibernate执行Save之前来做,那么看似拦截器和监听器都能实现,但实践一下就会知道,如果我们用监听器来做(比如此时用PreInsertEventListener),当我们在此改变某属性值后,会发现,保存入库的仍然是原来的值,监听器的Pre******是不允许我们这样做的。此时就需要用到拦截器,实现其OnSave方法,在此进行处理。
说完了这些,就简单说下他们的用法及配置方法。
一、
首先说监听器,我们以PostUpdateEventListener讲解
新建一个我们自己的监听器类,实现PostUpdateEventListener接口即可
- public class MyListenerextends DefaultLoadEventListener
- implements PostUpdateEventListener {
- public void onPostUpdate(PostUpdateEvent event) {
- System.out.println(event.getEntity().getClass().getName()+":更新完毕");
- for (int i =0; i < event.getState().length; i++) {
- // 更新前的值
- Object oldValue = event.getOldState()[i];
- // 更新后的新值
- Object newValue = event.getState()[i];
- //更新的属性名
- String propertyName = event.getPersister().getPropertyNames()[i];
- }
- }
- }
- public class MyListener extends DefaultLoadEventListener
- implements PostUpdateEventListener {
- public void onPostUpdate(PostUpdateEvent event) {
- System.out.println(event.getEntity().getClass().getName()+":更新完毕");
- for (int i = 0; i < event.getState().length; i++) {
- // 更新前的值
- Object oldValue = event.getOldState()[i];
- // 更新后的新值
- Object newValue = event.getState()[i];
- //更新的属性名
- String propertyName = event.getPersister().getPropertyNames()[i];
- }
- }
- }
public class MyListener extends DefaultLoadEventListener
implements PostUpdateEventListener {
public void onPostUpdate(PostUpdateEvent event) {
System.out.println(event.getEntity().getClass().getName()+":更新完毕");
for (int i = 0; i < event.getState().length; i++) {
// 更新前的值
Object oldValue = event.getOldState()[i];
// 更新后的新值
Object newValue = event.getState()[i];
//更新的属性名
String propertyName = event.getPersister().getPropertyNames()[i];
}
}
}
如果要实现其他监听器,只需实现其他的监听器接口即可,具体有哪些监听器,大家可以查看Hibernate包下的org.hibernate.event,里面能看到所有的监听器。
之后要加载监听器,我这里只讲解用过spring注入sessionFactory,用property配置的方法,因为另外的两种配置方法在网上很容易搜索到,这里就不重复。
先将我们的监听器定义到spring中
- <bean id="myListener"class="com.app.common.util.hibernateSupport.MyListener">
- <bean id="myListener" class="com.app.common.util.hibernateSupport.MyListener">
<bean id="myListener" class="com.app.common.util.hibernateSupport.MyListener">
在sessionFactory的bean标签中加入如下内容:
- class="org.springframework.orm.hibernate3.LocalSessionFactoryBean">
- ...
- <propertyname="eventListeners">
- <map>
- <entrykey="post-update">
- <refbean="myListener"/>
- </entry>
- </map>
- </property>
- ...
- </bean>
- <bean id="sessionFactory"
- class="org.springframework.orm.hibernate3.LocalSessionFactoryBean">
- ...
- <property name="eventListeners">
- <map>
- <entry key="post-update">
- <ref bean="myListener" />
- </entry>
- </map>
- </property>
- ...
- </bean>
<bean id="sessionFactory"
class="org.springframework.orm.hibernate3.LocalSessionFactoryBean">
...
<property name="eventListeners">
<map>
<entry key="post-update">
<ref bean="myListener" />
</entry>
</map>
</property>
...
</bean>
这样我们的监听器就可以工作了,当Hibernate执行update操作时,就会输出我们的信息了。
二、
再来说说拦截器
拦截器的实现要比监听器简单得多。
我们只要新建一个自己的拦截器类,继承自org.hibernate.EmptyInterceptor类,重写原来的方法,之后在spring中配置即可。
简单例子:
- @Override
- public boolean onSave(Object entity, Serializable id, Object[] state,
- String[] propertyNames, Type[] types) {
- //entity就是当前的实体对象
- //如果当前操作的TbUser,则做处理
- if(entity instanceof TbUser){
- TbUser user=(TbUser)entity;
- user.setUserPassword("123");
- if(user.getUserId()==null){
- //返回true则拦截本次操作
- return true;
- }
- }
- return super.onSave(entity, id, state, propertyNames, types);
- }
- }
- public class MyIntercept extends EmptyInterceptor {
- @Override
- public boolean onSave(Object entity, Serializable id, Object[] state,
- String[] propertyNames, Type[] types) {
- //entity就是当前的实体对象
- //如果当前操作的TbUser,则做处理
- if(entity instanceof TbUser){
- TbUser user=(TbUser)entity;
- user.setUserPassword("123");
- if(user.getUserId()==null){
- //返回true则拦截本次操作
- return true;
- }
- }
- return super.onSave(entity, id, state, propertyNames, types);
- }
- }
public class MyIntercept extends EmptyInterceptor {
@Override
public boolean onSave(Object entity, Serializable id, Object[] state,
String[] propertyNames, Type[] types) {
//entity就是当前的实体对象
//如果当前操作的TbUser,则做处理
if(entity instanceof TbUser){
TbUser user=(TbUser)entity;
user.setUserPassword("123");
if(user.getUserId()==null){
//返回true则拦截本次操作
return true;
}
}
return super.onSave(entity, id, state, propertyNames, types);
}
}
配置文件:
- <bean id="myIntercept"
- class="com.app.common.util.hibernateSupport.MyIntercept">
- .......
- <bean id="sessionFactory"
- class="org.springframework.orm.hibernate3.LocalSessionFactoryBean">
- ...
- <propertyname="entityInterceptor">
- <refbean="myIntercept"/>
- </property>
- ...
- </bean>
- <bean id="myIntercept"
- class="com.app.common.util.hibernateSupport.MyIntercept">
- .......
- <bean id="sessionFactory"
- class="org.springframework.orm.hibernate3.LocalSessionFactoryBean">
- ...
- <property name="entityInterceptor">
- <ref bean="myIntercept"/>
- </property>
- ...
- </bean>