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

被折叠的 条评论
为什么被折叠?



