- 浏览: 203765 次
- 性别:
- 来自: 成都
文章分类
最新评论
-
liouyan9:
jiangshiwei2008 写道老是提醒Servlet s ...
spring-mvc入门(一):入门实例 -
hochoy:
...
spring-mvc入门(二):后端控制器(上) -
springmvc_springjpa:
spring mvc demo教程源代码下载,地址:http: ...
spring-mvc入门(一):入门实例 -
yahier:
想问,不在代码中去try catch的话 log4j怎么记录项 ...
SSH集成log4j日志环境 -
13_木木:
谢谢分享,明白了很多
spring-mvc入门(一):入门实例
3.继承关系:
在前面的部门员工实例中,我们设定的员工只是普通的员工,现在假如有Sale和Skill两类员工,它们作为Employee的子类。如何实现这些子类的映射?为了方便以后的操作说明和不影响以前的操作,我们把前面用到的员工部门类及相关实体配置文件,主配置文件等拷贝到一个新的项目theExtend下进行操作。
(1)共享一张表:
意思是我们把子类的当作父类来处理共同映射成一张表。
>>步骤一,创建Employee的子类:Sale和Skill。内容如下:
package com.asm.hibernate.domain; public class Sale extends Employee { private String signSale; ...省略getXXX setXXX() } package com.asm.hibernate.domain; public class Skill extends Employee{ private String signSkill; ...省略getXXX setXXX() }
>>步骤二、修改Employee.hbm.xml配置文件:
<hibernate-mapping package="com.asm.hibernate.domain"> <class name="Employee" discriminator-value="0"> <id name="id"> <generator class="native" /> </id> <discriminator column="sign" type="string" /> <property name="name"></property> <many-to-one name="depart" column="depart_id" /> <subclass name="Sale" discriminator-value="1"> <property name="signSale"></property> </subclass> <subclass name="Skill" discriminator-value="2"> <property name="signSkill"></property> </subclass> </class> </hibernate-mapping>
配置文件说明: 理解<subclass>元素:它作为class的子元素,但是它却和class非常相似,name同样是指所关联到的“类”,下面的<property>也与class下的<property>一样。我们注意到它和class均增加了一个新属性discriminator-value,它的作用就是配置这些类的“鉴别类型”。而discriminator-value 的配置主要参照的是<discriminator>,它通常称为鉴别类型,即是说鉴别属于什么类型,因为在涉及到继承关系的操作时,总会涉及到父子类的关系,比如在上面的配置中,我们设定了sign子段来标识类型,执行后会再对此作说明。注意:<discriminator>要写在后面元素的前面,因为它的dtd文件是这样规定得。
>>步骤三、修改主配置文件,由于是拷贝的前面的文件,所以需要去掉无关的映射文件,否则会提示找不到这此映射文件。
>>步骤四、编写测试类:
package com.asm.hibernate.test; public class ManyToOneTest { public static void main(String[] args) { add(); Employee emp=query(2); System.out.println("emp type:"+emp); } static Employee query(int empId) { Session s = null; try { s = HibernateUtil.getSession(); Employee emp = (Employee) s.get(Employee.class, empId); System.out.println("Department Name:" + emp.getDepart().getName()); return emp; } finally { if (s != null) s.close(); } } static void add() { Session s = null; Transaction tx = null; try { Department depart = new Department(); depart.setName("departName"); Employee emp = new Employee(); emp.setName("empName"); emp.setDepart(depart); Sale emp2 = new Sale(); emp2.setName("saleEmployee"); emp2.setSignSale("saleName"); emp2.setDepart(depart); Skill emp3 = new Skill(); emp3.setName("skillEmployee"); emp3.setSignSkill("skillName"); emp3.setDepart(depart); s = HibernateUtil.getSession(); tx = s.beginTransaction(); s.save(emp); s.save(emp2); s.save(emp3); s.save(depart); tx.commit(); } finally { if (s != null) s.close(); } } }
说明:没的什么可多言的,只是要注意在查询时能返回其子类型。
下面来看执行后employee表的内容:
+----+------+---------------+-----------+----------+-----------+
| id | sign | name | depart_id | signSale | signSkill |
+----+------+---------------+-----------+----------+-----------+
| 1 | 0 | empName | 1 | NULL | NULL |
| 2 | 1 | saleEmployee | 1 | saleName | NULL |
| 3 | 2 | skillEmployee | 1 | NULL | skillName |
+----+------+---------------+-----------+----------+-----------+
先来看sign这列:由于sign是鉴别类型设定的字段,且分别在前面为Employee、Sale、Skill分别配置了“0、1、2”所以它们会在sign体现出来。其实<discriminator column="sign" type="string"/>也可以把type属性值设为“int”等。
再来看“signSale、signSkill”字段:它们本身是专为特定的类的属性配置的字段(比如在第一个<subclass>元素下的property子元素就配置了Sale类的signSale属性):所以它们只适合特定的类,而不适的类将会以null来填充,这也就是如果采取“共享一张表”的最大缺点,它限制了我们不能在子类属性所映射的字段上设定“非空”。由于查询只涉及到一张表,所以效率较高。
(2)每个子类一张附表:
意思是每个类均会有一张表,但是它不是完整的表,因为它的一些字段还在父类的表中。即是说:公共字段放在父表中,子类子类分别放在子类所映射的表中,它们之间采取主外键关联。这样解决了上面的“不能设定为空”的缺限。接上面只需修改Employee.hbm.xml配置文件,修改后的内容如下:
<class name="Employee"> <id name="id"> <generator class="native" /> </id> <property name="name"></property> <many-to-one name="depart" column="depart_id" /> <joined-subclass name="Sale"> <key column="sale_id" /> <property name="signSale" /> </joined-subclass> <joined-subclass name="Skill"> <key column="skill_id" /> <property name="signSkill" /> </joined-subclass> </class>
配置文件说明:当每个子类都会有一张表,在class子元素下设定了<joined-subclass>元素,它的意思就是专门指定为子类映射成一张表,特别要说明的是我们为每个子类都配置了<key>元素,它的作用就是作为外键关联employee表,它的值也是参照class元素的id来生成。执行后的表内容如下: skill表 +----------+-----------+ | skill_id | signSkill | +----------+-----------+ | 3 | skillName | +----------+-----------+ sale表 +---------+----------+ | sale_id | signSale | +---------+----------+ | 2 | saleName | +---------+----------+
employee表
+----+---------------+-----------+
| id | name | depart_id |
+----+---------------+-----------+
| 1 | empName | 1 |
| 2 | saleEmployee | 1 |
| 3 | skillEmployee | 1 |
+----+---------------+-----------+
执行后请留意hibernate产生的sql语句。
(3)联合使用表:
意思是同时使用(1)(2)的形式,主要目的是为了能使用鉴别类型,但同时也能“设定为非空”。同样只需要修改配置文件,修改后的内容如下:
<class name="Employee" discriminator-value="0"> <id name="id"> <generator class="native" /> </id> <discriminator column="sign" type="string" /> <property name="name"></property> <many-to-one name="depart" column="depart_id" /> <subclass name="Sale" discriminator-value="1"> <property name="signSale"></property> </subclass> <subclass name="Skill" discriminator-value="2"> <join table="skill"> <key column="skill_id" /> <property name="signSkill" /> </join> </subclass> </class>
观察上面的配置文件,不难发现我们对Sale采取了(1)方式,对Skill采取了(3)方式。同样请留意hibernate的sql语句和执行后的表。特别要说明的是:在进行操作时,由于hibernate在进行操作时不能删除前面的相关联的表(主要是和第一种继承关系“共享一张表”时发生冲突,因为在建立的三张表,在未删除sale/skill表时,是不能来删除employee表,因为employee表中的主键被另两张表关联了),所以需要手工删除表或者是直接删除数据库再建数据库。 补充:借助此例我们来看看discriminator-value的默认情况:如果我们在<subclass name="Skill" discriminator-value="2">中不写discriminator-value配置,再来执行发现一切正常,查表发现它的sign值为“com.asm.hibernate.domain.Skill”即完整的类名,属于字串,所以执行正常。但当我们把<discriminator>中的type属性改成int时将会报错,因为默认不写是以完整的类名字串来作为标识,而我们设定type为int,所以将不能匹配类型。
(4)每个具体类一张完整表:
与第(2)种情况相比,它的主要特点就是为每个具体类建立一张表,表的字段对应子类本身的属性,同样也包括父类的所有属性。同样只需要修改配置文件如下:
<class name="Employee"> <id name="id"> <generator class="hilo" /> </id> <property name="name"></property> <many-to-one name="depart" column="depart_id" /> <union-subclass name="Sale"> <property name="signSale" /> </union-subclass> <union-subclass name="Skill"> <property name="signSkill" /> </union-subclass> </class>
注意:id生成器选择了“hilo”,由于我们为每个具体类都映射了一张表,所以id不能只在每张表中递增,如果只在每张表中递增,这里的三张表中的id将会出现重复,这样我们在采用多态查询,将会查出多种结果,我们应让id是唯一的,所以采取了hilo的方式来生成id,它能保证id是全局性的递增生成,这样每张表中的id均不会重复。 同样请注意,可能需要删除某些表或者是删库建库才能执行测试类。 执行完成后,需留意hibernate产生的sql语句和表的结构内容。 补充说明:在使用这种方法时,如果父类为抽象类也是可行得,我们可以在<class>元素中配置abstract=”true”来说明父类为抽象类,自然就不会为其建表。
总结:在上面的继续关系中我们多次用到了删库建库,在执行测试类时,如果出现sql不能更新或者sql相关的错误,则不防尝试此方法。另外在学继承关系时,除了注意配置文件外,更应注意hibernate产生的sql语句以及执行后产生表的情况。通常我们建议表的数目不要超过类的数目。
发表评论
-
原创hibernate入门笔记[共计46页]
2010-05-05 16:04 4374第三次分享自己的原创笔记,希望大家多多指正, 我 ... -
hiberante入门(十七):简要总结及源码文档
2009-10-17 17:04 1328十三、总结: 1.主配置与实体(映射)配置: 关于这些配置 ... -
hiberante入门(十六):一些细节理论
2009-10-17 17:03 1586十二、一些细节问题分析 1.实体配置文件中的类型问题: 在 ... -
hiberante入门(十五):事务相关-悲观乐观锁
2009-10-17 17:02 19261.事务引发的三层架构MVC讨论:在前面的程序代码中多次用到了 ... -
hiberante入门(十四):缓存
2009-10-17 17:01 19671.模拟缓存并简要说明缓存实现原理 在myhibernate ... -
hiberante入门(十三):懒加载
2009-10-17 17:01 14804.懒加载: 在前面我们已经对懒加载有所提及,现在再借助一个 ... -
hibernate入门(十一):级联操作inversin属性
2009-10-17 16:59 1302九、关联关系中的高级 ... -
hiberante入门(十):其它集合类型
2009-10-17 16:58 1274八、其它集合类型 说明:在前面我们已经使用过set集合类型, ... -
hibernate入门(九):组件关系映射
2009-10-06 13:02 12465.组件关系映射: 典型实例:每个人有不同的名字,或者多个人 ... -
hibernate入门(八):多对多
2009-10-03 12:38 15064.多对多关系映射: 典型实例:一个学生可以有多个老师, ... -
hibernate入门(七):一对一
2009-10-03 12:33 16013.一对一关系映射: 典型的实例:一个人有一个身份 ... -
hibernate入门(六):一对多
2009-10-03 05:21 12842.一对多关系映射: 上面提到的多个员工对应于一个部门,是多 ... -
hibernate入门(五):多对一
2009-10-03 05:19 1462七 关联关系讨论 1.多对一关系映 ... -
hibernate入门(四):HQL QBC初步
2009-10-03 05:08 2147五 完善工具类及HQL QBC初步相关1.无聊的讨论: 在前 ... -
hibernate入门(三):对象的三种状态
2009-10-03 05:06 1831三 Session中的主要方法 ... -
hibernate入门(二):优化代码
2009-10-03 04:58 1413二 优化代码 1.为会么要优化 在前面我们已经知道,获取S ... -
hibernate入门(一)
2009-10-03 04:48 1727一 第一个应用实例 1.搭建环境:新建一个名为Hiberna ...
相关推荐
在本篇博文中,我们将深入探讨Hibernate框架中的一个重要特性——一对一(One-to-One)关系映射。Hibernate作为Java领域中最流行的ORM(对象关系映射)工具,它允许开发者以面向对象的方式处理数据库操作,简化了...
NULL 博文链接:https://86asm.iteye.com/blog/493165
hiberante3 注解帮助文档hiberante3 注解帮助文档hiberante3 注解帮助文档hiberante3 注解帮助文档hiberante3 注解帮助文档
在Java的持久化框架中,Hibernate是一个非常流行的ORM(对象关系映射)工具,它允许开发者将数据库操作转化为对Java对象的操作。DAO(Data Access Object)层是软件设计模式中的一个重要概念,主要用于处理数据访问...
Hibernate,作为一款强大的对象关系映射(ORM)框架,深受Java开发者喜爱。本文将结合"hibernate源码"和"hibernate配置"两个核心主题,深入探讨Hibernate的工作原理及配置细节,帮助读者从源码层面提升对Hibernate的...
最新springboot2基础hiberante5完整项目,打包jar,运行jsp,包括后台与前台,拦截器,登录,后台下载就可以使用,注意不是jpa,里面有完整Dao,千万级数据项目分离的代码,为了适合老项目开发特意集成hiberante5....
Hibernate 是一个强大的对象关系映射(ORM)框架,它简化了数据库操作,将数据库交互抽象为 Java 对象。 在“spring+mvc+hibernate”整合中,我们首先会设置 Spring 的配置,这通常涉及到定义 beans 和他们的依赖...
Hibernate,作为一个强大的对象关系映射(ORM)框架,是Java开发者在处理数据库操作时的得力助手。本篇将深入探讨Hibernate3.jar及其API,帮助开发者更好地理解和运用这个库。 一、Hibernate3.jar介绍 Hibernate3....
开发工具:MyEclipse 6....Struts+Spring+Hiberante框架整合的简单登录系统 无需配置任何文件、只需在mysql中创建一个空数据库 如:create database test; 注:mysql数据库用户名:root 密码:root
Hibernate是一个强大的Java持久化框架,它为开发者提供了在Java应用程序中管理关系数据库模型的工具。在5.0.7版本中,Hibernate引入了一系列改进和增强,使得它在处理数据库操作时更加高效和便捷。 **一、Hibernate...
Hibernate Tools是一套全新而且完整的面向Hibernate3的工具集合,它包含了Eclipse插件和Ant编译流程。Hibernate Tools是JBoss Tools的核心组件,所以他也是JBoss Developer Studio的一部分
Spring MVC 和 Hibernate 是两个在Java Web开发中广泛使用的框架,它们分别是用于构建MVC(Model-View-Controller)架构的Web应用和管理数据库操作的对象关系映射(ORM)工具。在实际项目中,这两个框架的集成能提供...
hiberante4.2.3-part2
在Java世界中,Hibernate是一个非常流行的ORM(对象关系映射)框架,它允许开发者使用面向对象的方式来操作数据库,而无需关心底层SQL语句的编写。本文将深入探讨Hibernate的四种主要查询方式:HQL(Hibernate Query...
### Hibernate中的五大核心接口 #### 一、概述 在Hibernate框架中,存在五大核心接口,它们分别是:`Session`、`SessionFactory`、`Transaction`、`Query` 和 `Configuration`。这些接口对于Hibernate的正常运行至...
标题“Hiberante3相关文档”表明了主要讨论的是关于Hibernate3这一持久化框架的资料集合,可能涵盖了多个方面,如查询语言、缓存机制以及数据加载策略。 描述中的“Hiberante3_HQL”提示我们将会涉及到Hibernate...
【标题】"job4j_hiberante"是一个与学习Hibernate相关的课程作业,它可能是Job4J平台的一个教学项目,旨在帮助学员深入理解并熟练运用Hibernate这一强大的Java对象关系映射(ORM)框架。 【描述】提到的是"课程作业...
hiberante-4.2.3-part3
- `@Inheritance`: 控制实体类继承时的表策略,如单表继承、独立表继承等。 - `@Transient`: 标记不持久化的属性。 4. **持久化类的创建** 使用Hibernate注解,开发者可以在Java类上直接标注数据库相关的元数据...
hiberante-4.2.3-part4