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. poj 3790 Recursively Palindromic Partitions (递推)

    题目 题意:求输入的数字的递归回文. 思路:答案等于这个数字一半之前的所有的 之和. #include <iostream> #include <cstdio> #includ ...

  2. 发布 windows 10 universal app 时微软账号验证失败

    具体错误:Visual Studio encountered an unexpected network error and can't contact the Microsoft account s ...

  3. Codeforces Round #270

    A 题意:给出一个数n,求满足a+b=n,且a+b均为合数的a,b 方法一:可以直接枚举i,n-i,判断a,n-i是否为合数 #include<iostream> #include< ...

  4. python模拟http请求2

    发现了一个非常好用的第三方module:requests,模拟接口非常简单. 详细了解请移步:http://docs.python-requests.org/en/latest/ 非常不错 #!cod ...

  5. 如何让你的 Asp.Net Web Api 接口,拥抱支持跨域访问。

    由于 web api 项目通常是被做成了一个独立站点,来提供数据,在做web api 项目的时候,不免前端会遇到跨域访问接口的问题. 刚开始没做任何处理,用jsonp的方式调用 web api 接口, ...

  6. php通过curl调用jpush接口实现消息的推送

    public function actionNotifyto() { //$regid = $_REQUEST['regid']; $url = 'https://api.jpush.cn/v3/pu ...

  7. HDU 4607 Park Visit (DP最长链)

    [题目]题意:N个城市形成一棵树,相邻城市之间的距离是1,问访问K个城市的最短路程是多少,共有M次询问(1 <= N, M <= 100000, 1 <= K <= N). [ ...

  8. directdraw的多画面显示rgb

    // showpicDlg.cpp : 实现文件 // #include "stdafx.h" #include "showpic.h" #include &q ...

  9. poj 3694 Network

    题意: 添加每条新连接后网络中桥的数目// 超时 先放着了 ,下次改//早上这代码超时了 下午改了,代码在下面#include <iostream> #include <algori ...

  10. Android 如何使用juv-rtmp-client.jar向Red5服务器发布实时视频数据

    使用juv-client-client.jar主要是尽快地完成毕业设计里面手机端向网页端发送实时视频的功能,由于实习和做毕业设计的时间冲突,因此完成毕业设计只花了1个多月时间. (万恶的形式主义,论文 ...