hibernate one-to-many关联查询<set>中使用filter

本文介绍了在Hibernate中处理one-to-many关联查询时,如何在`set`中使用`filter`进行条件过滤。通过示例展示了查询特定id College及其关联的满足条件的学生,强调了在`College.hbm.xml`中声明过滤器的不合理性,指出应该由`Student.hbm.xml`定义属性类型。同时,提到了`<set>`的`fetch`属性设置为`join`会导致过滤器失效,而`select`或`subselect`则能支持过滤。最后指出,虽然`filter`能实现关联过滤,但使用HQL、SQL或Criteria查询方式可能会更简便。

假如有一个College实体和Student实体,两者之间是one-to-many的关系,一个大学可以有多个学生,但是一个学生只能属于一个大学。参考hibernate manual可以很容易的配出一个one-to-many的单向关联。

package onemany.model;

import java.util.Set;

public class College {

	private int id;

	private String name;

	private Set<Student> allStudents;

	.....
}
package onemany.model;

public class Student {

	private int sid;

	private String sname;

	......
}

<?xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC
        "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
        "http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd">

<hibernate-mapping>
	<class name="onemany.model.College" table="t_association_college">
		<id name="id" column="cid" type="int">
			<generator class="native" />
		</id>

		<property name="name" column="cname" type="string" />

		<!-- set的fetch除了selec|join,还增加了一个subselect选项. subselect,选发出一条sql查询实体对象,再发出一条sql查询关联的实体对象. -->
		<set name="allStudents" table="t_association_student" fetch="subselect"
			lazy="false">
			<key column="college_id" not-null="true" />
			<one-to-many class="onemany.model.Student" />
		</set>
	</class>
</hibernate-mapping>
<?xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC
        "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
        "http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd">

<hibernate-mapping>
	<class name="onemany.model.Student" table="t_association_student">
		<id name="sid" column="sid" type="int">
			<generator class="native" />
		</id>

		<property name="sname" column="sname" type="string" />
	</class>
	
</hibernate-mapping>


下面这段代码,我们查询出id=2的College,会级联查出该college下面的所有学生。

@Test
public void testLoad() {
	Session session = sessionFactory.openSession();

	College college = (College) session.load(College.class, 2);

	System.out.println("college name is:" + college.getName());

	System.out.println(college.getAllStudents().size());

	session.close();
}



上面这段代码会级联查询所有的学生,但是不能过滤出符合特定条件的学生。我们可以使用filter来过滤。修改java代码很简单,打开filter即可。
@Test
public void testLoad() {
	Session session = sessionFactory.openSession();

	session.enableFilter("nameLikeFilter").setParameter("name", "%qun%");

	College college = (College) session.load(College.class, 2);

	System.out.println("college name is:" + college.getName());

	System.out.println(college.getAllStudents().size());

	session.close();
}


现在我们修改College的实体映射文件如下:
<?xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC
        "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
        "http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd">

<hibernate-mapping>
	<class name="onemany.model.College" table="t_association_college">
		<id name="id" column="cid" type="int">
			<generator class="native" />
		</id>

		<property name="name" column="cname" type="string" />

		<!-- set的fetch除了selec|join,还增加了一个subselect选项. subselect,选发出一条sql查询实体对象,再发出一条sql查询关联的实体对象. -->
		<set name="allStudents" table="t_association_student" fetch="subselect"
			lazy="false">
			<key column="college_id" not-null="true" />
			<one-to-many class="onemany.model.Student" />
			
			<filter name="nameLikeFilter">
		    <![CDATA[
		    	sname like :name
		    ]]>
			</filter>
			
		</set>
	</class>
	
	<filter-def name="nameLikeFilter">
		<filter-param name="name" type="string" />
	</filter-def>
	
</hibernate-mapping>


发出的SQL查询条件如下:可以看到过滤器起作用了。




我们完全是在College.hbm.xml中声明和引用过滤器的,这其实有点不合理。我们是想按照name模糊查询的,但是name的类型到底是什么,其实应该放在Student.hbm.xml中定义。只有自己的实体才知道自己属性的类型。

<?xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC
        "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
        "http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd">

<hibernate-mapping>
	<class name="onemany.model.College" table="t_association_college">
		<id name="id" column="cid" type="int">
			<generator class="native" />
		</id>

		<property name="name" column="cname" type="string" />

		<!-- set的fetch除了selec|join,还增加了一个subselect选项. subselect,选发出一条sql查询实体对象,再发出一条sql查询关联的实体对象. -->
		<set name="allStudents" table="t_association_student" fetch="subselect"
			lazy="false">
			<key column="college_id" not-null="true" />
			<one-to-many class="onemany.model.Student" />
			
			<filter name="nameLikeFilter">
		    <![CDATA[
		    	sname like :name
		    ]]>
			</filter>
			
		</set>
	</class>
	
</hibernate-mapping>
<?xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC
        "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
        "http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd">

<hibernate-mapping>
	<class name="onemany.model.Student" table="t_association_student">
		<id name="sid" column="sid" type="int">
			<generator class="native" />
		</id>

		<property name="sname" column="sname" type="string" />
	</class>
	
	<filter-def name="nameLikeFilter">
		<filter-param name="name" type="string" />
	</filter-def>
</hibernate-mapping>

我们将<filter-def>放在了Student的实体映射文件中,然后在College的实体映射文件<set>中引用过滤器。经过我的验证,这样也是可以的。由此可以看出:<filter-def>是一个全局效果,无论放在哪儿个实体映射文件,效果都是一样的


有一点需要特别注意:<set>的fetch有3个可选值:select、join、subselect。如果将fetch设置成join,那么过滤器失效。设置成select和subselect都是可以过滤的


可以看到:虽然是以filter的确实现了关联关系的过滤,但是很麻烦,而且有很大的局限性:fetch不能使用join。如果使用HQL、SQL或者Criteria查询,可以更简单地实现过滤。


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值