我们都知道,Hibernate最大的一个优点就是使开发更加“面向对象”,类与类之间有继承关系,Hibernate中也对这种继承关系提供了映射的封装。
Hibernate为继承映射提供了三种策略
1、每棵继承树使用一张表
2、每个子类使用一张表
3、每个具体类使用一张表
本文对第一种策略进行说明。
场景
如下类图
上图中Pig类和Bird类继承Animal类,每棵继承树对应一张表,即在同一棵继承树中,所有的类的对象信息(记录)共同存放到一张表中,要判断某条记录属于哪个对象,需要在表中添加一个字段进行区分(比如下表的Type字段)。
(表 1)
配置
PO对象
Animal.java
public class Animal { private int id; private String name; private boolean sex; //getter、setter }
Bird.java
public class Bird extends Animal{ private int height; //getter、setter }
Pig.java
public class Pig extends Animal{ private int weight; //getter、setter }
映射文件
配置映射文件时,父类还用<class>标签来定义即可;添加的区分字段(比如上面表1中的Type字段)需要用<discriminator>标签来定义;用<subclass>标签定义两个子类,与父类“合并”在同一张表里,子类的特有属性用<property>属性定义即可。
Extends.hbm.xml
<hibernate-mapping package="com.danny.hibernate"> <class name="Animal"> <id name="id"> <generator class="native"/> </id> <discriminator column="type" type="string"></discriminator> <property name="name"/> <property name="sex"/> <subclass name="Bird" discriminator-value="B"> <property name="height"></property> </subclass> <subclass name="Pig" discriminator-value="P"> <property name="weight"></property> </subclass> </class> </hibernate-mapping>
映射文件中的<subclass>
标签还可以与标签同级,但是要加上属性extends,属性值为父类全路径名称。
启动程序执行的建表语句为:
drop table if exists Animal create table Animal (id integer not null auto_increment, type varchar(255) not null, name varchar(255), sex bit, height integer, weight integer, primary key (id))
插入测试
session=HibernateUtils.getSession(); session.beginTransaction(); Pig pig=new Pig(); pig.setName("小猪猪"); pig.setSex(true); pig.setWeight(200); session.save(pig); Bird bird=new Bird(); bird.setName("小鸟鸟"); bird.setSex(false); bird.setHeight(100); session.save(bird); Animal animal=new Animal(); animal.setName("小动物"); animal.setSex(false); session.save(animal); session.getTransaction().commit();
插入结果为:
插入父类(Animal)时,默认把类名当做type了
查询测试
load查询
根据配置,鉴别值(表中的type)在存储的时候会自动存储,在加载的时候也会根据鉴别值映射取得相应的对象。
比如查询id为1的那条数据,既可以用Pig查询,也可以用Animal查询。
用session.load(Pig.class, 1)查询:
session.beginTransaction(); Pig pig=(Pig)session.load(Pig.class, 1); System.out.println(pig.getName()); session.getTransaction().commit();
用session.load(Animal.class, 1)查询:
session.beginTransaction(); Animal pig=(Animal)session.load(Animal.class, 1); System.out.println(pig.getName()); session.getTransaction().commit();
执行结果都为:
小猪猪
如果用load方法查询的话,默认是不支持多态查询(hibernate在加载数据的时候会自动鉴别类的真正类型)的,因为load默认支持lazy(懒加载),所以上面的pig只是Animal的代理,因此用instanceof也就判断不出来pig的类型,如下:
session=HibernateUtils.getSession(); session.beginTransaction(); Animal animal=(Animal)session.load(Animal.class, 1); if(animal instanceof Pig){ System.out.println(animal.getName()); }else if(animal instanceof Bird){ System.out.println(animal.getName()); }else{ System.out.println("既不是小猪猪也不是小鸟鸟"); } session.getTransaction().commit();
运行结果为:
既不是小猪猪也不是小鸟鸟
想要支持多态查询也简单,在配置文件中标签后加lazy=”false”即可,禁止懒加载就OK了。
get查询
get查询支持多态查询:
session=HibernateUtils.getSession(); session.beginTransaction(); Animal animal=(Animal)session.get(Animal.class, 1); if(animal instanceof Pig){ System.out.println(animal.getName()); }else if(animal instanceof Bird){ System.out.println(animal.getName()); }else{ System.out.println("既不是小猪猪也不是小鸟鸟"); } session.getTransaction().commit();
运行结果:
既不是小猪猪也不是小鸟鸟
hql查询
session=HibernateUtils.getSession(); session.beginTransaction(); List animalList=session.createQuery("from Animal").list(); for(Iterator iter=animalList.iterator();iter.hasNext();){ Animal animal=(Animal)iter.next(); System.out.println(animal.getName()); } session.getTransaction().commit();
总结
这种映射方式可以把多个类放在一张表中,但是粒度比较粗,有冗余字段;但又是因为多个类的相关记录都存放在一张表中,查询时不用关联,因此效率较高。