前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Java中使用Hibernate系列之过滤器(filters)学习

Java中使用Hibernate系列之过滤器(filters)学习

作者头像
用户1289394
发布2018-02-27 15:26:31
9420
发布2018-02-27 15:26:31
举报
文章被收录于专栏:Java学习网

Hibernate3新增了对某个类或者集合使用预先定义的过滤器条件(filter criteria)的功能。过滤器条件相当于定义一个 非常类似于类和各种集合上的“where”属性的约束子句,但是过滤器条件可以带参数。 应用程序可以在运行时决定是否启用给定的过滤器,以及使用什么样的参数值。 过滤器的用法很像数据库视图,只不过是在应用程序中确定使用什么样的参数的。

网络配图

要使用过滤器,必须首先在相应的映射节点中定义。而定义一个过滤器,要用到位于<hibernate-mapping/> 节点之内的<filter-def/>节点:

代码语言:javascript
复制
<filter-def name="myFilter"><filter-param name="myFilterParam" type="string"/></filter-def>

定义好之后,就可以在某个类中使用这个过滤器:

代码语言:javascript
复制
<class name="myClass" ...>...<filter name="myFilter" condition=":myFilterParam = MY_FILTERED_COLUMN"/></class>

也可以在某个集合使用它:

代码语言:javascript
复制
<set ...><filter name="myFilter" condition=":myFilterParam = MY_FILTERED_COLUMN"/></set>

可以在多个类或集合中使用某个过滤器;某个类或者集合中也可以使用多个过滤器。

Session对象中会用到的方法有:enableFilter(String filterName), getEnabledFilter(String filterName), 和 disableFilter(String filterName). Session中默认是不启用过滤器的,必须通过Session.enabledFilter()方法显式的启用。 该方法返回被启用的Filter的实例。以上文定义的过滤器为例:

代码语言:javascript
复制
session.enableFilter("myFilter").setParameter("myFilterParam", "some-value");

注意,org.hibernate.Filter的方法允许链式方法调用。(类似上面例子中启用Filter之后设定Filter参数这个“方法链”) Hibernate的其他部分也大多有这个特性。

下面是一个比较完整的例子,使用了记录生效日期模式过滤有时效的数据:

代码语言:javascript
复制
<filter-def name="effectiveDate"><filter-param name="asOfDate" type="date"/></filter-def><class name="Employee" ...>...<many-to-one name="department" column="dept_id" class="Department"/><property name="effectiveStartDate" type="date" column="eff_start_dt"/><property name="effectiveEndDate" type="date" column="eff_end_dt"/>...<!--Note that this assumes non-terminal records have an eff_end_dt set toa max db date for simplicity-sake注意,为了简单起见,此处假设雇用关系生效期尚未结束的记录的eff_end_dt字段的值等于数据库最大的日期--><filter name="effectiveDate"condition=":asOfDate BETWEEN eff_start_dt and eff_end_dt"/></class><class name="Department" ...>...<set name="employees" lazy="true"><key column="dept_id"/><one-to-many class="Employee"/><filter name="effectiveDate"condition=":asOfDate BETWEEN eff_start_dt and eff_end_dt"/></set></class>

定义好后,如果想要保证取回的都是目前处于生效期的记录,只需在获取雇员数据的操作之前先开启过滤器即可:

代码语言:javascript
复制
Session session = ...;session.enabledFilter("effectiveDate").setParameter("asOfDate", new Date());List results = session.createQuery("from Employee as e where e.salary > :targetSalary").setLong("targetSalary", new Long(1000000)).list();

在上面的HQL中,虽然我们仅仅显式的使用了一个薪水条件,但因为启用了过滤器,查询将仅返回那些目前雇用 关系处于生效期的,并且薪水高于一百万美刀的雇员的数据。

注意:如果你打算在使用外连接(或者通过HQL或load fetching)的同时使用过滤器,要注意条件表达式的方向(左还是右)。 最安全的方式是使用左外连接(left outer joining)。并且通常来说,先写参数, 然后是操作符,最后写数据库字段名。

在Filter定义之后,它可能被附加到多个实体和/或集合类,每个都有自己的条件。假若这些条件都是一样的,每次都要定义就显得很繁琐。因此,<filter-def/>被用来定义一个默认条件,它可能作为属性或者CDATA出现:

代码语言:javascript
复制
<filter-def name="myFilter" condition="abc > xyz">...</filter-def><filter-def name="myOtherFilter">abc=xyz</filter-def>

当这个filter被附加到任何目的地,而又没有指明条件时,这个条件就会被使用。注意,换句话说,你可以通过给filter附加特别的条件来重载默认条件。

本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2017-12-21,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 Java学习网 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档