一对多关联关系映射(one to many+单向)

一对多关联关系映射(one to many)
注意:
load和get是根据主键来加载数据的,不是主键是加载不出来的。




hihernate一对多关联映射(单向Classes----->Student)

一对多关联映射利用了多对一关联映射原理

多对一关联映射:在多的一端加入一个外键指向一的一端,它维护的关系是多指向一
一对多关联映射:在多的一端加入一个外键指向一的一端,它维护的关系是一指向多

也就是说一对多和多对一的映射策略是一样的,只是站的角度不同

在一一端维护关系的缺点:
* 如果将t_student表里的classesid字段设置为非空,则无法保存
* 因为不是在student这一端维护关系,所以student不知道是哪个班的,
 所以需要发出多余的update语句来更新关系



在对象模型中,一对多的关联关系,使用集合来表示
比如Classes(班级)和Student(学生)之间是一对多的关系,我们就以班级和学生为例来说明,班级需要持有学生的一个集合。



第一步:我们首先建立实体关系类:如下
班级的;
public class Classes {
private int id;
private String name;
private Set students; //在class类里持有一个学生的集合,使用set是因为set里面的东西是不能重复的
public int getId() {
return id;
}

public void setId(int id) {
this.id = id;
}

public String getName() {
return name;
}

public void setName(String name) {
this.name = name;
}

public Set getStudents() {
return students;
}

public void setStudents(Set students) {
this.students = students;
}
}
学生的:

public class Student {
private int id;
private String name;

public int getId() {
return id;
}

public void setId(int id) {
this.id = id;
}

public String getName() {
return name;
}

public void setName(String name) {
this.name = name;
}
}


第二步是建立实体的映射文件:
student的映射文件是:
<hibernate-mapping>
<class name="com.bjsxt.hibernate.Student" table="t_student">
<id name="id">
<generator class="native"/>
</id>
<property name="name"/>
</class>
</hibernate-mapping>

class的映射文件是:
<hibernate-mapping package="com.bjsxt.hibernate">
<class name="Classes" table="t_classes">
<id name="id">
<generator class="native"/>
</id>
<property name="name"/>

_____________________________________________________________________________________________________________________________
//关键就在这:集合使用set标签来声明,name就是属性名。上面声明的就是students。
<set name="students">
<key column="classesid"/>//key标签,给一个列的名字,就是加一个字段,他就会把classid这个字段加入到student表里。做为外键指向student,注意这个字段是加到student表里了
<one-to-many class="Student"/>//采用one-to-many标签一方面指明是一对多关系映射,另一方面采用class属性指明这个集合里是什么类型的元素,这里是
//Student类型的元素,平常需要加包路径,因为在package里写过了,故这不用写完整路径
</set>
___________________________________________________________________________________________________________________________

</class>
</hibernate-mapping>



第三步写hibernate的配置文件:
<hibernate-configuration>
<session-factory>
<property name="hibernate.connection.url">jdbc:mysql://localhost/hibernate_one2many_1</property>
<property name="hibernate.connection.driver_class">com.mysql.jdbc.Driver</property>
<property name="hibernate.connection.username">root</property>
<property name="hibernate.connection.password">bjsxt</property>
<property name="hibernate.dialect">org.hibernate.dialect.MySQLDialect</property>
<property name="hibernate.show_sql">true</property>
<mapping resource="com/bjsxt/hibernate/Classes.hbm.xml"/>
<mapping resource="com/bjsxt/hibernate/Student.hbm.xml"/>
</session-factory>
</hibernate-configuration>






第四步:写测试代码:
注意,我们必须先得有学生,然后把学生放到集合里,故先建学生。
public class One2ManyTest extends TestCase {

public void testSave1() {
Session session = null;
try {
session = HibernateUtils.getSession();
session.beginTransaction();
//先建学生
Student student1 = new Student();
student1.setName("10");
session.save(student1);//学生需要先存上,这个时候classid没有值。
Student student2 = new Student();
student2.setName("祖儿");
session.save(student2);//这个时候classid也是没值
Set students = new HashSet();//采用HashSet
students.add(student1);
students.add(student2);
Classes classes = new Classes();
classes.setName("尚学堂");
classes.setStudents(students);
//可以正确保存
session.save(classes);//这个时候classid有值了
session.getTransaction().commit();//要连续发出俩个update语句去更新student表中的classid值。从这可以看出,classid在开始的时候是空的值
//如果我们先把classid属性设为非空,则就会出问题。就存不进去。所以这样是有确定的。再个当有多条数据时,要发出多条update语句,这比较耗时。
}catch(Exception e) {
e.printStackTrace();
session.getTransaction().rollback();
}finally {
HibernateUtils.closeSession(session);
}
}
public void testLoad1() {
Session session = null;
try {
session = HibernateUtils.getSession();
session.beginTransaction();
Classes classes = (Classes)session.load(Classes.class, 1);
System.out.println("classes.name=" + classes.getName());
Set students = classes.getStudents();
for (Iterator iter=students.iterator(); iter.hasNext();) {
Student student = (Student)iter.next();
System.out.println("student.name=" + student.getName());
}
session.getTransaction().commit();
}catch(Exception e) {
e.printStackTrace();
session.getTransaction().rollback();
}finally {
HibernateUtils.closeSession(session);
}
}
}




































posted @ 2009-06-03 22:17  刘阳  阅读(315)  评论(0编辑  收藏  举报