Hibernate学习笔记(二)

第一章 单向一对多关联

1. 怎么实现一对多关联?

  • 在数据库中,可以通过添加主外键的关联,表现一对多的关系
  • 在Hibernate中,通过在一方持有多方的集合来实现一对多的关系,即在“一”的一端中使用set集合元素表示持有“多”的一端的对象

2.具体步骤(手动配置MyEclipse)

2.1 导入相应的jar包、创建核心配置文件

    A:添加jar包:java项目下建立lib文件夹->把jar包拷贝进去(大概9个)->选中jar包右击->build path->add to path
    B:从Hibernate解压缩包中随便拷贝一个hibernate.cfg.xml文件到src目录->修改相应配置即可

hibernate.cfg.xml

<hibernate-configuration>
    <session-factory>
        <property name="hibernate.dialect">org.hibernate.dialect.MySQLDialect</property>
        <property name="hibernate.connection.driver_class">com.mysql.jdbc.Driver</property>
        <property name="hibernate.connection.username">root</property>
        <property name="hibernate.connection.password">123</property>
        <property name="hibernate.connection.url">
            <![CDATA[
                jdbc:mysql://localhost:3306/hibernate2?useUnicode=true&characterEncoding=utf8
            ]]>
        </property>
        <property name="show_sql">true</property>
        <property name="hbm2ddl.auto">update</property>

        <!-- 指定映射文件的路径 -->
        <mapping resource="com/imooc/entity/Grade.hbm.xml"/>
        <mapping resource="com/imooc/entity/Student.hbm.xml"/>

    </session-factory>
</hibernate-configuration>

2.2 创建数据表(学生表和班级表),创建持久化类(并且将其序列化),分别和数据库中的表相对应

Grade .java

//持久化类
public class Grade implements Serializable {
    private int gid;
    private String gname;
    private String gdesc;
    // 在一对多的一方定义一个多方的集合
    private Set<Student> students = new HashSet<Student>();

    public Grade(String gname, String gdesc) {
        super();
        this.gname = gname;
        this.gdesc = gdesc;
    }
    public Grade(int gid, String gname, String gdesc, Set<Student> students) {
        super();
        this.gid = gid;
        this.gname = gname;
        this.gdesc = gdesc;
        this.students = students;
    }
    public Grade() {
        super();
    }
    ...

Student.java

//持久化类
public class Student implements Serializable {
    private static final long serialVersionUID = 1L;
    private int sid;
    private String sname;
    private String sex;
    // 单向多对一配置(两个步骤)
    // 1.在多方定义一个一方的引用
    private Grade grade;
    //2.修改多方映射文件的配置
    //<many-to-one name="grade" class="com.imooc.entity.Grade" column="gid"></many-to-one>
    public Student(String sname, String sex) {
        super();
        this.sname = sname;
        this.sex = sex;
    }

2.3 为持久化类创建对应的映射文件(在Hibernate解压缩包下随便拷贝一个,做修改),然后在hibernate.cfg.xml中指定映射文件的路径(一定不能忘记)这样学生到班级的单向一对多配置就完成了

Grade.hbm.xml

<hibernate-mapping>
    <class name="com.imooc.entity.Grade" table="grade">
    <!-- name表示主键对应的属性名称,column表示主键在数据库对应的列的名称 -->
        <id name="gid" column="gid" type="java.lang.Integer">
            <!-- 主键生成策略 -->
            <generator class="increment"></generator>
        </id>
        <property name="gname" type="java.lang.String">
            <column name="gname" length="20" not-null="true"></column>
        </property>
        <property name="gdesc">
            <column name="gdesc"></column>
        </property>
        <!-- 配置单向一对多关联关系 inverse="true"表示由多方维护关联关系  inverse="true" cascade="save-update"-->
        <set name="students" table="student" >
        <!-- 指定关联的外键 -->
            <key column="gid"></key>
            <one-to-many class="com.imooc.entity.Student"/>
        </set>
    </class>
</hibernate-mapping>

这里写图片描述
Student.hbm.xml

<hibernate-mapping>
    <class name="com.imooc.entity.Student" table="student">
        <id name="sid" column="sid" type="java.lang.Integer">
            <!-- 主键生成策略 -->
            <generator class="increment"></generator>
        </id>
        <property name="sname" type="java.lang.String">
            <column name="sname" length="20" not-null="true"></column>
        </property>
        <property name="sex">
            <column name="sex"></column>
        </property>
        <!-- 配置多对一关联关系 
        <many-to-one name="grade" class="com.imooc.entity.Grade"  column="gid"></many-to-one>-->
    </class>
</hibernate-mapping>

3. 对一对多关联关系进行测试

3.1 创建session工具类

HibernateUtil.java

//session工具类
public class HibernateUtil {

    private static SessionFactory sessionFactory;
    private static Session session;

    //用静态代码快来进行初始化
    static{
        //创建Configuration对象,读取hibernate.cfg.xml文件,完成初始化
        Configuration config = new Configuration().configure();
        StandardServiceRegistryBuilder ssrb = new StandardServiceRegistryBuilder()
            .applySettings(config.getProperties());
        StandardServiceRegistry ssr = ssrb.build();
        sessionFactory = config.buildSessionFactory(ssr);
    }

    //获取SessionFactory
    public static SessionFactory getSessionFactory() {
        return sessionFactory;
    }

    //获取Session
    public  static Session getSession() {
        session = sessionFactory.openSession();
        return session;
    }

    //关闭Session
    public static void closeSession(Session session) {
        if (session!=null) {
            session.close();
        }
    }
}

3.2 创建测试类

Test.java

/*
 * 单向多对一关系(班级-->学生)
 * 建立关联关系后,可以方便的从一个对象导航到另一个对象
 * 注意关联的方向(目前不能通过学生找到学生所在的班级)
 */
public class Test {
    public static void main(String[] args) {
         //add();
         findStudentByGrade();
        // update();
        //delete();
    }

    // 将学生添加到班级
    public static void add() {
        Grade g = new Grade("java一班", "java软件开发一班");
        Student stu1 = new Student("张三", "男");
        Student stu2 = new Student("穆女神", "女");

        // 如果希望学生在学生表中添加对应的班级编号,需要在班级中添加学生,建立关联关系
        g.getStudents().add(stu1);
        g.getStudents().add(stu2);

        Session session = HibernateUtil.getSession();
        Transaction tx = session.beginTransaction();
        session.save(g);
        session.save(stu1);
        session.save(stu2);
        tx.commit();
        HibernateUtil.closeSession(session);
    }

    // 查询班级中包含的学生(可以找到班级表中没有包含的学生详细信息)
    public static void findStudentByGrade() {
        Session session = HibernateUtil.getSession();
        Grade grade = (Grade) session.get(Grade.class, 1);
        System.out.println(grade.getGname() + "," + grade.getGdesc());

        Set<Student> students = grade.getStudents();
        for (Student student : students) {
            System.out.println(student.getSname() + "," + student.getSex());
        }

    }

    // 修改学生信息(将学生1的班级修改为2班)
    public static void update() {
        Grade g = new Grade("java二班", "java软件开发二班");

        Session session = HibernateUtil.getSession();
        Transaction tx = session.beginTransaction();
        Student stu = (Student) session.get(Student.class, 1);
        g.getStudents().add(stu);
        session.save(g);
        tx.commit();
        HibernateUtil.closeSession(session);
    }

    // 删除学生信息
    public static void delete() {
        Session session = HibernateUtil.getSession();
        Transaction tx = session.beginTransaction();
        Student stu = (Student) session.get(Student.class, 2);
        session.delete(stu);
        tx.commit();
        HibernateUtil.closeSession(session);
    }

}

第二章 单向多对一关联

  • 多对一的关系和关系数据库中的外键参照关系最匹配,即在己方的表中的一个外键参照另一个表的主键
  • 通过在多方持有一方的引用实现,需要在“多”的一端使用many-to-one配置

1. 在多方持久化类(Student.java)中添加一个一方的引用

2. 在多方映射文件(Student.hbm.xml)中配置一对多关联关系

3. 同时配置单向多对一和单向一对多关联关系就变成了双向多对一关联关系了

3. 测试双向多对一关联关系

// 测试保存方法
    public static void save() {
        Grade g = new Grade("java一班", "java软件开发一班");
        Student stu1 = new Student("穆女神", "女");
        Student stu2 = new Student("小穆穆", "男");

        // 设置关联关系
        // 班级到学生(一对多)向班级添加学生
        g.getStudents().add(stu1);
        g.getStudents().add(stu2);
        // 学生到班级(多对一)为学生添加班级
        stu1.setGrade(g);
        stu2.setGrade(g);

        Session session = HibernateUtil.getSession();
        Transaction tx = session.beginTransaction();
        session.save(g);// 级联操作
        // session.save(stu1);
        // session.save(stu2);
        tx.commit();
        HibernateUtil.closeSession(session);
    }

测试查询

// 查询学生所在的班级信息(这些信息并没有在学生表中保存)
    public static void findGradeByStudent() {
        Session session = HibernateUtil.getSession();
        Student stu = (Student) session.get(Student.class, 2);
        System.out.println(stu.getSid() + "," + stu.getSname() + ","
                + stu.getSex());
        //得到班级,就可以查询班级的信息了
        Grade g = stu.getGrade();
        System.out
                .println(g.getGid() + "," + g.getGname() + "," + g.getGdesc());
        HibernateUtil.closeSession(session);

    }

4. inverse属性

在进行双向多对一测试时,可以发现执行的sql语句为:

Hibernate: select max(gid) from grade
Hibernate: select max(sid) from student
Hibernate: insert into grade (gname, gdesc, gid) values (?, ?, ?)
Hibernate: insert into student (sname, sex, gid, sid) values (?, ?, ?, ?)
Hibernate: insert into student (sname, sex, gid, sid) values (?, ?, ?, ?)
Hibernate: update student set gid=? where sid=?
Hibernate: update student set gid=? where sid=?

后面的两条update语句是是一方在维护一对多的关系(更新学生所在的班级),而多方(Student)知道自己所属的班级,所以这两条sql语句是多余的。可以在Grade.hbm.xml的set属性中设置inverse值为true表示由多方维护关联关系。如此,后面两条sql语句便不会再执行。

Grade.hbm.xml

<set name="students" table="student"  inverse="true">
        <!-- 指定关联的外键 -->
            <key column="gid"></key>
            <one-to-many class="com.imooc.entity.Student"/>
        </set>

这里写图片描述

5. cascade属性

在进行双向多对一测试时,我们既保存了班级信息,又保存了学生信息。班级是知道自己班有哪些学生的,所以当我们保存班级时,正常来说应该自动保存班级的学生信息才对。所以需要设置级联属性。

<set name="students" table="student"  inverse="true" cascade="save-update">

这样,测试方法就只需保存班级就可以了

        session.save(g);
        // 级联操作(无需再次显示的保存学生信息)
        //session.save(stu1);
        //session.save(stu2);

这里写图片描述

第三章 使用MyEclipse自动配置Hibernate环境和关联关系

  1. 创建数据库连接:新建java项目,打开MyEclipse数据库视图,按照提示建立一个数据库连接
  2. 添加Hibernate支持:选中项目->右击->MyEclipse->add hibernate capabilities->一直next即可
    要注意的两点:添加数据库链接和新建一个工具包存放自己生成的获得session的工具类类
    此时MyEclipse为我们项目导入了jar包、创建了核心配置文件、创建了获取session的工具类
  3. MyEclipse反向工程:根据数据库中的表自动生成实体化类和映射文件,并且可以选择一对多和多对一关联关系;

    进入数据库视图,选中要配置的表->右键选中reverse engineering->配置相关选项即可
    
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值