Hibernate的事件监听机制

Hibernate中的事件监听机制可以对Session对象的动作进行监听,一旦发生了特殊的事件,Hibernate就会执行监听器中的事件处理方法

在某些功能的设计中,我们即可以使用Hibernate的拦截器实现,也可以使用Hibernate的事件监听来实现

Hibernate中事件与对应的监听器接口

org.hibernate.event. EventListeners

事件类型 监听器接口

auto-flush---- -------------AutoFlushEventListener

merge----------------------MergeEventListener

create----------------------PersistEventListener

delete----------------------DeleteEventListener

dirty-check----------------DirtyCheckEventListener

evict------------------------EvictEventListener

flush------------------------FlushEventListener

flush-entity----------------FlushEntityEventListener

load-------------------------LoadEventListener

load-collection-------------InitializeCollectionEventListener

lock--------------------------LockEventListener

refresh----------------------RefreshEventListener

replicate--------------------ReplicateEventListener

save-update---------------SaveOrUpdateEventListener

save------------------------SaveOrUpdateEventListener

update---------------------SaveOrUpdateEventListener

pre-load-------------------PreLoadEventListener

pre-update---------------PreUpdateEventListener

pre-delete----------------PreDeleteEventListener

pre-insert-----------------PreInsertEventListener

post-load-----------------PostLoadEventListener

post-update--------------PostUpdateEventListener

post-delete---------------PostDeleteEventListener

post-insert----------------PostInsertEventListener

post-commit-update-----PostUpdateEventListener

post-commit-delete------PostDeleteEventListener

post-commit-insert-------PostInsertEventListener

应用Hibernate事件监听器

用户制定的事件监听器首先需要实现与所需要处理的事件对应的接口,或者继承实现这个接口的类

通过使用Hibernate的配置文件(hibernate.cfg.xml)配置事件监听对象,或者使用Configuration对象注册这个定制的事件监听器对象

LogPostLoadEventListener

import org.hibernate.event.PostLoadEvent;

import org.hibernate.event.PostLoadEventListener;

public class LogPostLoadEventListener implements PostLoadEventListener {

private static final long serialVersionUID = 404241098418965422L;

public void onPostLoad(PostLoadEvent event) {

System.out.println("Class:" + event.getEntity().getClass().getName() + ",id:"

+ event.getId());

}

}

使用监听器实现审计日志

利用Hibernate的事件机制,不仅能够精确追踪到持久化对象的字段的修改,持久化对象关联关系的变更,还能记录更新前的数值和更新后的数值

拦截器(Interceptor)
org.hibernate.Interceptor接口定义了Hibernate中通用拦截机制
创建Session对象的时候,所有的Session对象或者这个Session对象的所有持久化操作的动作都会被指定的拦截器进行拦截.

Interceptor接口的方法

afterTransactionBegin()
当一个事务时候启动时,会立刻调用这个方法,这个方法可以改变这个事务的状态,例如:回滚事务

instantiate()
创建对象,如果返回null,则Hibernate将调用实体类的默认构造方法创建持久化对象

getEntity()
当一个持久化对象,通过标示符属性在Session对象的缓存中进行查找,并且没有找到时,会调用该方法

getEntityName()
当session对象获取持久化对象的名字时,会调用这个方法

onLoad()
该方法在持久化对象初始化之前加载,这个的持久化对象处于刚被创建的状态(对象的属性值都未赋值)

findDirty()
当调用Session对象的flush()方法时,讲调用该方法判断对象是否为脏数据,这是脏数据检查的另外拦截的实现方式

isTransient()
当调用Session对象的saveOrUpdate方法时,会调用该方法判断对象是否尚未保存

onSave()
在对象被保存之前调用,通过这个方法可以对要保持的对象的属性进行修改

onDelete()
该方法在持久化对象被删除之前调用

preFlush()
该方法当调用Session对象的flush()方法之前被调用

onFlushDirty()
当调用Session对象flush()方法进行脏数据检查时,如果发现持久化对象的状态发生了改变,会调用该方法

postFlush()
该方法调用Session对象的flush()方法之后被调用

beforeTransactionCompletion()
在完成一个事务之前,调用此方法,这个方法可以改变事务的状态,例如回滚事务

afterTransactionCompletion()
当完成一个事务之后,立刻调用此方法

使用拦截器实现审计日志
审计日志指的是,在应用系统中,对所有的数据库的操作都做记录,记录所操作内容,操作的用户和操作的时间

demo

log4j.properties

Log4j.properties代码

log4j.logger.com.rbh.examples=info,appender1

log4j.appender.appender1=org.apache.log4j.FileAppender

log4j.appender.appender1.layout=org.apache.log4j.TTCCLayout

log4j.appender.appender1.File=ligfile.txt

LogEntityInterceptor

package com.rbh.examples;

import java.io.Serializable;

import org.apache.log4j.Logger;

import org.hibernate.EmptyInterceptor;

import org.hibernate.type.Type;

public class LogEntityInterceptor extends EmptyInterceptor {

private static final long serialVersionUID = 1L;

private Logger logger = Logger.getLogger(LogEntityInterceptor.class);

public void onDelete(Object entity,Serializable id, Object[] state,String[] propertyNames,

Type[] types){

logger.info("删除数据");

}

public boolean onFlushDirty(Object entity,Serializable id, Object[] currentState,

Object[] preState,String[] propertyNames,

Type[] types){

logger.info("修改数据");

return false;

}

public boolean onSave(Object entity,Serializable id, Object[] State,

String[] propertyNames,

Type[] types){

logger.info("保存数据");

return false;

}

}

HibernateTest

package com.rbh.examples;

import java.util.Date;

import org.hibernate.Session;

import org.hibernate.SessionFactory;

import org.hibernate.cfg.Configuration;

public class HibernateTest {

public static void main(String[] args)

{

HibernateTest test =new HibernateTest();

test.testInterceptor();

}

public void testInterceptor()

{

LogEntityInterceptor interceptor=new LogEntityInterceptor();

Configuration config=new Configuration();

config.setInterceptor(interceptor);

config.configure();

SessionFactory sf=config.buildSessionFactory();

Session session=sf.getCurrentSession();

Guestbook gb= new Guestbook();

gb.setName("Narcissus");

gb.setEmail("javac.q@gmail.com");

gb.setCreatedTime(new Date());

gb.setPhone("11102121");

gb.setTitle("test Interceptor");

gb.setContent("test Interceptor,test Interceptor");

session.beginTransaction();

session.save(gb);

session.getTransaction().commit();

session=sf.getCurrentSession();

gb.setName("tom");

session.beginTransaction();

session.update(gb);

session.getTransaction().commit();

session=sf.getCurrentSession();

session.beginTransaction();

session.delete(gb);

session.getTransaction().commit();

}

}

可以通过session方式加载拦截器对象,也可以通过Configuration对象加载拦截器

Configuration:对所有的session都会被拦截
session:只对当前的session进行拦截

监听器与拦截器的比较

监听器可以实现更细化粒度的拦截
通过监听器获取所拦截的持久化对象的修改后喝修改前的状态值
能直接通过Event对象获取Session对象

Hibernate监听器的更多相关文章

  1. 利用Hibernate监听器实现用户操作日志

    网上搜索发现,实现用户操作日志的方式有:自定义注解方式.Hibernate拦截器方式.Hibernate监听器方式等. 1.自定义注解方式较为麻烦,需要进行操作记录的方法均需要添加注解,但是相对的操作 ...

  2. hibernate监听器的应用

    这里是我看到的一个hibernate监听器的简单实现供参考  http://www.360doc.com/content/14/0623/11/8072791_389034447.shtml 设计思路 ...

  3. AES实现财务数据的加密解密存储

    需求背景 众所周知,金融行业有各种各样的财务报表,有些报表涉及到公司财务或经营相关的敏感数据,需要进行加密存储,只有掌握密钥的用户才能看到解密后的数据.注意,这里所说的加密并不是针对整个数据库或者表全 ...

  4. Hibernate拦截器(Interceptor)与事件监听器(Listener)

    拦截器(Intercept):与Struts2的拦截器机制基本一样,都是一个操作穿过一层层拦截器,每穿过一个拦截器就会触发相应拦截器的事件做预处理或善后处理. 监听器(Listener):其实功能与拦 ...

  5. SSH项目web.xml文件的常用配置【struts2的过滤器、spring监听器、解决Hibernate延迟加载问题的过滤器、解决中文乱码的过滤器】

    配置web.xml(struts2的过滤器.spring监听器.解决Hibernate延迟加载问题的过滤器.解决中文乱码的过滤器) <!-- 解决中文乱码问题 --> <filter ...

  6. hibernate的拦截器和监听器

    拦截器(Intercept):顾名思义,拦截操作,也就是在Hibernate做出动作之前会调用的方法.如果你有需要在Hibernate操作数据库之前想要做的操作,就需要用到这个东西了. 监听器(Lis ...

  7. 深入浅出Struts2+Spring+Hibernate框架

    一.深入浅出Struts2 什么是Struts2? struts2是一种基于MVC的轻量级的WEB应用框架.有了这个框架我们就可以在这个框架的基础上做起,这样就大大的提高了我们的开发效率和质量,为公司 ...

  8. Struts2+Spring+Hibernate框架整合总结详细教程

    一.SSH三大框架知识总结 Struts 2是Struts的下一代产品,是在 struts 1和WebWork的技术基础上进行了合并的全新的Struts 2框架.其全新的Struts 2的体系结构与S ...

  9. Hibernate(4)——主键生成策略、CRUD 基础API区别的总结 和 注解的使用

    俗话说,自己写的代码,6个月后也是别人的代码……复习!复习!复习!涉及的知识点总结如下: hibernate的主键生成策略 UUID 配置的补充:hbm2ddl.auto属性用法 注解还是配置文件 h ...

随机推荐

  1. Android adb 常用技巧

    1.在命令行管理模拟器设备(AVD) list:列出机器上所有已经安装的Android版本和AVD设备 list avd:列出机器上所有已经安装的AVD设备: list target:列出机器上所有已 ...

  2. bzoj1601: [Usaco2008 Oct]灌水

    经典延伸最小生成树问题... #include<cstdio> #include<cstring> #include<cctype> #include<alg ...

  3. Asp.net 后台添加Meta标签方法

    Asp.net 后台添加Meta标签方法包括keywords,CSS.JS 下面是从Asp.net 后台添加CSS.JS.Meta标签的写法,我们这里写成函数方便以后使用.如果函数放在页面类中, Pa ...

  4. WebService 出现因 URL 意外地以“/HelloWorld”结束,请求格式无法识别。

    要在webservice的web.config文件中的 <system.web> 节点下加入: <webServices>    <protocols>       ...

  5. 【转】那些不能错过的Xcode插件 -- 不错不错

    原文网址:http://www.cocoachina.com/industry/20130918/7022.html 古人云“工欲善其事必先利其器”,打造一个强大的开发环境,是立即提升自身战斗力的绝佳 ...

  6. java运用FFMPEG视频转码技术

    基于windows系统安装FFMPEG转码技术 http://wenku.baidu.com/link?url=z4Tv3CUXxxzLpa5QPI-FmfFtrIQeiCYNq6Uhe6QCHkU- ...

  7. C# C/S 结构操作Ini系统文件

    Winfrom 开发时,有时会将一些系统某个设置保存到Ini 类型的文件中.下面提供操作Ini 文件的代码: public static class IniFiles { [DllImport(&qu ...

  8. Nginx源码安装及调优配置

    导读 由于Nginx本身的一些优点,轻量,开源,易用,越来越多的公司使用nginx作为自己公司的web应用服务器,本文详细介绍nginx源码安装的同时并对nginx进行优化配置. Nginx编译前的优 ...

  9. 1050 数的计数 c语言实现

    描述 给定一个正整数,求其各位之和. 输入 输入一行,为一个正整数(最多10,000位). 输出 输出各位之和. 样例输入 17 样例输出 8 解析:这题主要是大数计算的问题,因为10000位的数无法 ...

  10. 【原】Storm序列化

    5. Storm高级篇 序列化 Dynamic typing Custom serialization Java serialization Component-specific serializat ...