Hibernate Inteceptor

在项目收尾阶段,客户突然提出一个很麻烦但也很正常的需求,系统内的一切改动都要进行记录。格式如: 
    2004.1.1 12:30  郁也风 销售订单 订货日期 2004.1.2->2004.1.3 
   
    第一时间我就想到了也很AOP的Trigger,但Trigger毕竟和Java代码是两个世界,怎么把操作员名字传进去就有点麻烦(要所有表再加一个 last modifier的列,然后修改程序),同时Trigger一个很不够AOP的地方就是有多少个表就要写多少个Triger,有多少个列就要写多少列。 
    Hibernate的Community Area 刚好有一篇History Interceptor的文章,很AOP的解决了这个问题。 

1.Hibernate Inteceptor 概念 
  Inteceptor--拦截器的概念源自AOP,在2.1.6中文版的参考手册里面这样写 
  Interceptor接口提供从session到你的应用程序的回调方法,让你的程序可以在持久化对象保存/更改/删除或者装载的时候操作它的属性。典型代码: 

Java代码 
  1. public class AuditInterceptor implements Interceptor, Serializable   
  2. {  
  3.    public boolean onFlushDirty(Object entity,   
  4.                                 Serializable id,   
  5.                                 Object[] currentState,  
  6.                                 Object[] previousState,  
  7.                                 String[] propertyNames,  
  8.                                 Type[] types)   
  9.    {  
  10.   
  11.         if ( entity instanceof Auditable )   
  12.        {  
  13.             for ( int i=0; i < propertyNames.length; i++ )   
  14.             {  
  15.                 if ( "lastUpdateTimestamp".equals( propertyNames[i] ) )   
  16.                {  
  17.                     currentState[i] = new Date();  
  18.                     return true;  
  19.                 }  
  20.             }  
  21.         }  
  22.         return false;  
  23.     }  
  24. }  


当session被创建的时候,指定拦截器。 
Java代码 
  1. Session session = sf.openSession( new AuditInterceptor() );  



可见,Interceptor的基本模式是 
1.提供onLoad(),onSave(),onDelete,onFlushDirty()几个回调函数。 
   在创建session时向session指定Interceptor. 

2.每个回调函数的参数如下,已经足够作任何事情 
   1.Object entity   对象 
   2.Serializable id   对象的id 
   3.String[] propertyNames  属性的名称数组 
   4.Type[] types   属性的类型数组 
   5.Object[] previousState  更新前的属性值 
   6.Object[] currentState    更新后的属性值 

3.在函数内,通过if ( entity instanceof Auditable )判断是否目标对象类型. 

2.History Interceptor模式 
1.所有需要记录更改历史的对象implement Historizable接口。 
   History Interceptor根据 if ( entity instanceof Historizable)l来判断是否需要干活。 

2.History Interceptor有setUserName()函数。在程序中把操作员名字传入。 

3.创建HistoryEntry的POJO和hbm 
   包含如下的列 id,who,what,property,oldValue,newValue 

4.为所有需要纪录更改历史的对象分别创建一个History表,仅仅是表名不同,(或者用同一个表,复合主键?) 

5.为所有需要纪录更改历史的对象在POJO和hbm中加入historyEntry Set。 

3.剩下的问题 
  1.单层的对象已经获得完美的解决,但对于订单的子对象又如何解决呢? 

  2.因为使用Interceptor的session需要自行创建,与Spring的声明式事务管理如何结合? 

  3.需要把被更改的表名和列名翻译成中文。 

    其中问题1是Trigger更加无法完成的。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值