hibernate单向 多对一关联

一个班级可以有多个学生,一个学生只能属于一个班级。因此学生到班级就是典型的单向多对一关系。

数据库的特征:在学生表中加一个班级的外键(此外键参考班级的主键)

 

程序的结构:



 
类图:

 


 


Student.java

package com.org.model;

public class Student {
	private int id;
	private String name;
	private MyClass classes;
	
	public MyClass getClasses() {
		return classes;
	}
	public void setClasses(MyClass classes) {
		this.classes = classes;
	}
	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;
	}
	
	

}

 MyClass.java

package com.org.model;

public class MyClass {
	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.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 package="com.org.model">
	<class name="Student" table="student">
		<id name="id">
			<generator class="native"/>
		</id>
		<property name="name" column="name" not-null="true"/>
		
   <!-- 
   name指Student类中对应一方的属性名  cascade属性指定级联操作
   class指一方的类名    column中的name表示关联的外键    not-null表示此外键不能为空
   -->
		
		<many-to-one name="classes" cascade="save-update" class="MyClass" >
		       <column name="class_id" not-null="true"></column>
		</many-to-one>
<!-- many-to-one用来映射多对一,name表示对象中的属性名称 ,column用来表示外键的名称-->
 <!-- 
   当设置了cascade的时候,会自动完成关联,如果添加时没有关联对象,会自动创建一个关联对象
   最佳实践:如果没有特殊情况不要使用cascade,特别注意,可能使用cascade的地方一般都是一的
   一方进行删除时使用,特殊需求才会使用cascade的add,正常情况add方法都是应该有程序员完成
   添加 
  -->
	</class>


</hibernate-mapping>

 MyClass.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 package="com.org.model">
	<class name="MyClass" table="class">
		<id name="id">
			<generator class="native"/>
		</id>
		<property name="name" not-null="true"/>
		
	</class>


</hibernate-mapping>

hibernate.cfg.xml

 

<?xml version='1.0' encoding='utf-8'?>
<!DOCTYPE hibernate-configuration PUBLIC
        "-//Hibernate/Hibernate Configuration DTD 3.0//EN"
        "http://www.hibernate.org/dtd/hibernate-configuration-3.0.dtd">

<hibernate-configuration>

	<session-factory>

		<!-- Database connection settings -->
		<property name="connection.driver_class">com.mysql.jdbc.Driver</property>
		<property name="connection.url">jdbc:mysql://localhost/hbn</property>
		<property name="connection.username">root</property>
		<property name="connection.password">12345</property>

		<!-- SQL dialect -->
		<property name="dialect">org.hibernate.dialect.MySQLDialect</property>

		<!-- Enable Hibernate's automatic session context management -->
		<property name="current_session_context_class">thread</property>

		<!-- Disable the second-level cache  -->
		<property name="cache.provider_class">org.hibernate.cache.NoCacheProvider</property>

		<!-- Echo all executed SQL to stdout -->
		<property name="show_sql">true</property>

		<property name="format_sql">true</property>

		
		
		<mapping resource="com/org/model/MyClass.hbm.xml" />
		<mapping resource="com/org/model/Student.hbm.xml" />



	</session-factory>

</hibernate-configuration>

  

HibernateTest.java

package com.org.model.test;

import org.hibernate.Session;
import org.hibernate.Transaction;
import org.junit.Test;

import com.org.model.MyClass;
import com.org.model.Student;
import com.org.util.HibernateUtil;

public class HibernateTest {

	@Test
	public void testSave1() {
		Session session = null;
		Transaction tx = null;
		try {

			// 拿到session
			session = HibernateUtil.getSession();
			System.out.println(session);
			// 开启事务
			tx = session.beginTransaction();

			// 给实体赋值
                 //最佳实践:一定要先添加一的一方,之后在添加多的一方
			MyClass classes1 = new MyClass();
			classes1.setName("09002");
			Student st1 = new Student();
			st1.setName("at");
			st1.setClasses(classes1);

			Student st2 = new Student();
			st2.setName("bt");
			st2.setClasses(classes1);
			
			session.save(st1);
			session.save(st2);

			// 提交事务
			tx.commit();

		} catch (Exception e) {
			// 打印堆栈信息
			e.printStackTrace();
			// 事务回滚
			if (tx != null) {
				tx.rollback();
			}
		}
	}

}

 在保存学生的同时,班级的信息也被保存,因为配置了级联关系。单向的多对一关系由多的一方维护。

如果没有配置级联关系,执行测试程序时会出现以下异常:

org.hibernate.PropertyValueException: not-null property references a null or transient value: com.org.model.Student.classes
	at org.hibernate.engine.Nullability.checkNullability(Nullability.java:100)
	at org.hibernate.event.def.AbstractSaveEventListener.performSaveOrReplicate(AbstractSaveEventListener.java:312)
	at org.hibernate.event.def.AbstractSaveEventListener.performSave(AbstractSaveEventListener.java:203)
	at org.hibernate.event.def.AbstractSaveEventListener.saveWithGeneratedId(AbstractSaveEventListener.java:129)
	at org.hibernate.event.def.DefaultSaveOrUpdateEventListener.saveWithGeneratedOrRequestedId(DefaultSaveOrUpdateEventListener.java:210)

 错误的原因是:没有配置级联关系时,在保存学生的时候不会自动保存它所对应的班级的,这样该班级就不存在,而学生表中的外键要为非空,因此出现了该异常
 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值