hibernate

第13章 拦截器与事件(Interceptors and events)

应用程序能够响应Hibernate内部产生的特定事件是非常有用的。这样就允许实现某些通用的功能 以及允许对Hibernate功能进行扩展。

13.1.拦截器(Interceptors)

Interceptor接口提供了从会话(session)回调(callback)应用程序(application)的机制, 这种回调机制可以允许应用程序在持久化对象被保存、更新、删除或是加载之前,检查并(或)修改其属性。一个可能的用途,就是用来跟踪审核(auditing)信息。例如:下面的这个拦截器,会在一个实现了 Auditable接口的对象被创建时自动地设置createTimestamp属性,并在实现了Auditable接口的对象被更新时,同步更新lastUpdateTimestamp属性。

package org.hibernate.test;

import java.io.Serializable;

import java.util.Date;

import java.util.Iterator;

import org.hibernate.Interceptor;

import org.hibernate.type.Type;

public class AuditInterceptor implements Interceptor, Serializable {

private int updates;

private int creates;

public void onDelete(Objectentity,

Serializableid,

Object[]state,

String[]propertyNames,

Type[]types) {

// do nothing

}

public booleanonFlushDirty(Object entity,

Serializableid,

Object[] currentState,

Object[] previousState,

String[] propertyNames,

Type[] types) {

if ( entity instanceofAuditable ) {

updates++;

for ( int i=0; i <propertyNames.length; i++ ) {

if ("lastUpdateTimestamp".equals( propertyNames[i] ) ) {

currentState[i]= new Date();

return true;

}

}

}

return false;

}

public boolean onLoad(Objectentity,

Serializable id,

Object[]state,

String[]propertyNames,

Type[]types) {

return false;

}

public boolean onSave(Objectentity,

Serializable id,

Object[]state,

String[]propertyNames,

Type[]types) {

if ( entity instanceofAuditable ) {

creates++;

for ( int i=0;i<propertyNames.length; i++ ) {

if ("createTimestamp".equals( propertyNames[i] ) ) {

state[i] = newDate();

return true;

}

}

}

return false;

}

public void postFlush(Iteratorentities) {

System.out.println("Creations:" + creates + ", Updates: " + updates);

}

public void preFlush(Iteratorentities) {

updates=0;

creates=0;

}

...

}

创建会话(session)的时候可以指定拦截器。

Session session = sf.openSession( new AuditInterceptor() );

你也可以使用Configuration来设置一个全局范围的拦截器。

new Configuration().setInterceptor( new AuditInterceptor() );

13.2.事件系统(Event system)

如果需要响应持久层的某些特殊事件,你也可以使用Hibernate3的事件框架。 该事件系统可以用来替代拦截器,也可以作为拦截器的补充来使用。

基本上,Session接口的每个方法都有相对应的事件。比如LoadEvent,FlushEvent,等等(查阅XML配置文件 的DTD,以及org.hibernate.event包来获得所有已定义的事件的列表)。当某个方法被调用时,Hibernate Session会生成一个相对应的事件并激活所 有配置好的事件监听器。系统预设的监听器实现的处理过程就是被监听的方法要做的(被监听的方法所做的其实仅仅是激活监听器,“实际”的工作是由监听器完成的)。不过,你可以自由地选择实现 一个自己定制的监听器(比如,实现并注册用来处理处理LoadEvent的LoadEventListener接口), 来负责处理所有的调用Session的load()方法的请求。

监听器应该被看作是单例(singleton)对象,也就是说,所有同类型的事件的处理共享同一个监听器实例,因此监听器 不应该保存任何状态(也就是不应该使用成员变量)。

用户定制的监听器应该实现与所要处理的事件相对应的接口,或者从一个合适的基类继承(甚至是从Hibernate自带的默认事件监听器类继承, 为了方便你这样做,这些类都被声明成non-final的了)。用户定制的监听器可以通过编程使用Configuration对象 来注册,也可以在Hibernate的XML格式的配置文件中进行声明(不支持在Properties格式的配置文件声明监听器)。 下面是一个用户定制的加载事件(loadevent)的监听器:

public class MyLoadListener extends DefaultLoadEventListener {

// this is the single methoddefined by the LoadEventListener interface

public Object onLoad(LoadEventevent, LoadEventListener.LoadType loadType)

throwsHibernateException {

if (!MySecurity.isAuthorized( event.getEntityClassName(), event.getEntityId() ) ) {

throwMySecurityException("Unauthorized access");

}

return super.onLoad(event,loadType);

}

}

你还需要修改一处配置,来告诉Hibernate以使用选定的监听器来替代默认的监听器。

<hibernate-configuration>

<session-factory>

...

<listenertype="load" class="MyLoadListener"/>

</session-factory>

</hibernate-configuration>

看看用另一种方式,通过编程的方式来注册它。

Configuration cfg = new Configuration();

cfg.getSessionEventListenerConfig().setLoadEventListener( newMyLoadListener() );

通过在XML配置文件声明而注册的监听器不能共享实例。如果在多个<listener/>节点中使用 了相同的类的名字,则每一个引用都将会产生一个独立的实例。如果你需要在多个监听器类型之间共享监听器的实例,则你必须使用编程的方式来进行注册。

为什么我们实现了特定监听器的接口,在注册的时候还要明确指出我们要注册哪个事件的监听器呢?这是因为一个类可能实现多个监听器的接口。在注册的时候明确指定要监听的事件,可以让启用或者禁用对某个事件的监听的配置工作简单些。

13.3.Hibernate的声明式安全机制

通常,Hibernate应用程序的声明式安全机制由会话外观层(session facade)所管理。现在,Hibernate3允许某些特定的行为由JACC进行许可管理,由JAAS进行授权管理。 本功能是一个建立在事件框架之上的可选的功能。

首先,你必须要配置适当的事件监听器(event listener),来激活使用JAAS管理授权的功能。

<listener type="pre-delete"class="org.hibernate.secure.JACCPreDeleteEventListener"/>

<listener type="pre-update"class="org.hibernate.secure.JACCPreUpdateEventListener"/>

<listener type="pre-insert"class="org.hibernate.secure.JACCPreInsertEventListener"/>

<listener type="pre-load"class="org.hibernate.secure.JACCPreLoadEventListener"/>

接下来,仍然在hibernate.cfg.xml文件中,绑定角色的权限:

<grant role="admin" entity-name="User"actions="insert,update,read"/>

<grant role="su" entity-name="User"actions="*"/>

这些角色的名字就是你的JACC provider所定义的角色的名字。


第12章事务和并发

起始页

第14章批量处理(Batch processing)

http://docs.huihoo.com/hibernate/reference-v3_zh-cn/events.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值