Hibernate继承映射之每个继承层次一张表

1.概述
关系数据库的表之间不存在继承关系,为了将面向对象中的继承关系映射到关系数据库中,可使用三种映射策略:

1、每个继承层次一张表
2、每个具体类一张表
3、每个类一张表

2.每个继承层次一张表
即用一张表表示一个继承层次,因此需要增加一个字段用于区别

public class Person {
    private int id;
    private String name;
    private String sex;
    private int age;
}
public class Student extends Person {
    private String sno;
    private String school;
}
public class Worker extends Person{
    private String no;
    private double salary;
}

image.png

映射语法:

1、定义一个以父类命名的映射文件
2、<id>标记对后面添加<discriminator>标记,并指定column属性来定义鉴别字段
3、父类属性放在<class>标记对之间,子类属性放在<subclass>标记对之间
4、在各类标记中使用discriminator-value属性指定各类对应的鉴别字段的值

示例:

1、数据库连接
2、为项目添加Hibernate支持
3、创建持久化类
4、创建映射文件
5、修改hibernate配置文件
6、创建工具类获取Session实例
7、编写Dao接口及其实现类
8、测试

项目结构:


image.png

Person类:

public class Person {
    private int id;
    private String name;
    private String sex;
    private int age;

    public Person() {}

    public Person(String name, String sex, int age) {
        this.name = name;
        this.sex = sex;
        this.age = age;
    }

    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 String getSex() {
        return sex;
    }

    public void setSex(String sex) {
        this.sex = sex;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }
}

Worker类:

public class Worker extends Person{
    private String no;
    private double salary;

    public Worker() {}
    public Worker(String name, String sex, int age, String no, double salary) {
        super(name, sex, age);
        this.no = no;
        this.salary = salary;
    }

    public String getNo() {
        return no;
    }

    public void setNo(String no) {
        this.no = no;
    }

    public double getSalary() {
        return salary;
    }

    public void setSalary(double salary) {
        this.salary = salary;
    }
}

Student类:

public class Student extends Person {
    private String sno;
    private String school;

    public Student() {
    }

    public Student(String name, String sex, int age, String sno, String school) {
        super(name, sex, age);
        this.sno = sno;
        this.school = school;
    }

    public String getSno() {
        return sno;
    }

    public void setSno(String sno) {
        this.sno = sno;
    }

    public String getSchool() {
        return school;
    }

    public void setSchool(String school) {
        this.school = school;
    }
}

工具类:

public class HibernateUtil {
    private static SessionFactory sf;

    // 工具类不允许实例化
    private HibernateUtil() {
    }
    // 静态代码块,只执行一次,提高性能
    static {
        Configuration cf = new Configuration().configure();
        sf = cf.buildSessionFactory();
    }
    public static Session getSession() {
        Session session = sf.getCurrentSession();
        return session;
    }
}

Dao及其实现类:

public interface HibernateDao {
    public void addPerson(Person person);
    @SuppressWarnings("unchecked")
    public List getAllPerson(String hql);
}

public class HibernateDaoImpl implements HibernateDao {
    @Override
    public void addPerson(Person person) {
        Session session = HibernateUtil.getSession();
        Transaction tx = session.beginTransaction();
        try {
            session.save(person);
            tx.commit();
        } catch (Exception e) {
            if (null != tx) {
                tx.rollback();
            }
            e.printStackTrace();
        }
    }

    @Override
    @SuppressWarnings("unchecked")
    public List getAllPerson(String hql) {
        Session session = HibernateUtil.getSession();
        Transaction tx = session.beginTransaction();
        Query query = session.createQuery(hql);
        List persons = query.list();
        tx.commit();
        return persons;
    }
}

测试类:

public class HibernateTest {
    public static void main(String[] args) {
        Person person = new Person("张三", "男", 20);
        Worker worker = new Worker("李四", "女", 28, "P0001", 6000);
        Student student = new Student("王五", "女", 19, "SD001", "广金");
        HibernateDao hibernateDao = new HibernateDaoImpl();
        hibernateDao.addPerson(person);
        hibernateDao.addPerson(student);
        hibernateDao.addPerson(worker);
        String hql = "from Person";
        List<Person> persons = hibernateDao.getAllPerson(hql);
        System.out.println("========所有人员信息=======");
        for (Person p : persons) {
            System.out.println("姓名:" + p.getName() + " 性别:" + p.getSex() + " 年龄:" + p.getAge());
        }
        hql = "from Worker";
        List<Worker> workers = hibernateDao.getAllPerson(hql);
        System.out.println("========工人信息=======");
        for (Worker w : workers) {
            System.out.println("姓名:" + w.getName() + " 性别:" + w.getSex() + " 年龄:" + w.getAge() +
                    " 工号:" + w.getNo() + "薪水:" + w.getSalary());
        }
        hql = "from Student";
        List<Student> students = hibernateDao.getAllPerson(hql);
        System.out.println("========学生信息=======");
        for (Student s : students) {
            System.out.println("姓名:" + s.getName() + " 性别:" + s.getSex() + " 年龄:" + s.getAge() +
                    " 学号:" + s.getSno() + "学校:" + s.getSchool());
        }
    }
}

Person.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="这里写你的Person类所在的包">
    <!--ORM对象、关系数据库映射-->
    <class name="Person" table="person" discriminator-value="person">
        <id name="id" type="integer">
            <generator class="native"/>
        </id>
        <!--设置鉴别字段-->
        <discriminator column="type" type="string" length="15"/>
        <!--从父类继承下来的共享属性-->
        <property name="name" type="string" column="username" length="12" not-null="true"/>
        <property name="sex" type="string" length="1"/>
        <property name="age" type="integer"/>
        <subclass name="Student" discriminator-value="student">
            <!--子类新增属性-->
            <property name="sno" type="string" length="10"/>
            <property name="school" type="string" length="50"/>
        </subclass>
        <subclass name="Worker" discriminator-value="worker">
            <!--子类新增属性-->
            <property name="no" column="wno" type="string" length="10"/>
            <property name="salary" type="double"/>
        </subclass>
    </class>
</hibernate-mapping>

hibernate配置文件:

<?xml version='1.0' encoding='utf-8'?>
<!DOCTYPE hibernate-configuration PUBLIC
    "-//Hibernate/Hibernate Configuration DTD//EN"
    "http://www.hibernate.org/dtd/hibernate-configuration-3.0.dtd">
<hibernate-configuration>
  <session-factory>
    <property name="connection.driver_class">com.mysql.jdbc.Driver</property>
    <property name="connection.url">jdbc:mysql://localhost:3306/hibernate?useSSL=true</property>
    <property name="connection.username">root</property>
    <property name="connection.password">这里填你的密码</property>
    <!--连接池设置-->
    <property name="connection.pool_size">2</property>
    <!--数据库方言设置-->
    <property name="dialect">org.hibernate.dialect.MySQL8Dialect</property>
    <!--向控制台显示执行的SQL语句-->
    <property name="show_sql">false</property>
    <!--格式化SQL语句后输出-->
    <property name="hibernate.format_sql">true</property>
    <!--事务配置-->
    <property name="current_session_context_class">thread</property>
    <!--创建SessionFactory对象时自动创建数据库表,可取create、create-drop、update等值-->
    <property name="hibernate.hbm2ddl.auto">update</property>
    <!--配置映射文件-->
    <mapping resource="Person.hbm.xml"/>
  </session-factory>
</hibernate-configuration>

运行结果:


image.png

数据库中:


image.png

补充说明:需要先建立数据库hibernate,此项目为maven项目。也可以建立普通java项目或web项目。

一个继承层次一张表的优缺点如下:

1、最简单,执行效率最高(无需关联)
2、存在冗余字段,需要加入区分各类的字段;不允许子类属性为not null约束(允许为空)。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,033评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,725评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,473评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,846评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,848评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,691评论 1 282
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,053评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,700评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,856评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,676评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,787评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,430评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,034评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,990评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,218评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,174评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,526评论 2 343

推荐阅读更多精彩内容