Hibernate是一个开放源代码的对象关系映射框架,它对JDBC进行了非常轻量级的对象封装,它将POJO与数据库表建立映射关系,是一个全自动的orm框架,hibernate可以自动生成SQL语句,自动执行,使得Java程序员可以随心所欲的使用对象编程思维来操纵数据库。Hibernate可以应用在任何使用JDBC的场合,既可以在Java的客户端程序使用,也可以在Servlet/JSP的Web应用中使用,最具革命意义的是,Hibernate可以在应用EJB的J2EE架构中取代CMP,完成数据持久化的重任。Hibernate是一个持久层的ORM框架。

测试运行环境:
windows 7 64,MyEclipse 2017,Hibernate 5.0.7,JDK 1.8,JavaEE 6.0,Tomcat 8.0,MySQL 5.7

Hibernate的优点

  1. 简化了JDBC编程,简化了dao层编码工作。
  2. 修改实体类不需要修改源代码。
  3. 性能优越,轻量级框架。映射灵活性出色。支持多种关系型数据库,从一对一到多对多的各种复杂关系。

Hibernate的简单配置

准备阶段

  1. 官网下载对应版本的压缩包
  2. 配置Hibernate运行环境
    • 测试的时候对应数据库驱动包
    • 日志包log4j
    • hibernate必须包(required文件中的所有包)
  3. 创建好数据库表结构和对应的JavaBean(全部使用包装类,不使用基本数据类型)

Hibernate映射的配置文件

  1. 创建映射的配置文件

    • 默认名称:实体类名.hbm.xml
    • 默认位置:在JavaBean所在的包下
    • 引入约束:在xml配置文件中引入约束(引入的是hibernate3.0的dtd约束,不要引入4的约束)
      1
      2
      3
      <!DOCTYPE hibernate-mapping PUBLIC
      "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
      "http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd">
  2. 链接本地约束(没有网络的情况下)

    • 复制http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd
    • window
    • preferences
    • 搜索xml
    • 选择xml catalog
    • 点击add
    • KeyType:URI
    • Key:粘贴复制的地址
    • Location:选择本地的DTD约束的路径
    • 约束位置:hibernate-core-5.0.7.Final.jar\org\hibernate
  3. 编写映射的配置文件

    • 一个JavaBean对应一个实体类名.hbm.xml的配置文件

    • 标签:用来将类与数据库表建立映射关系

      • name:类的全路径
      • table:表名.(类名与表名一致,那么table属性也可以省略)
      • catalog:数据库的名称(一般省略)
    • 标签:用来将类中的属性与表中的主键建立映射,id标签就是用来配置主键的。

      • name:类中属性名
      • column:表中的字段名(如果类中的属性名与表中的字段名一致,那么column可以省略)
      • length:字段的长度(如果数据库已经创建好了,那么length可以不写。如果没有创建好,生成表结构时,length最好指定)
    • 标签:用来将类中的普通属性与表中的字段建立映射

      • name:同上
      • column:同上
      • length:同上
      • type:数据类型(一般都不需要编写,如果写需要按着规则来编写)
        • Hibernate的数据类型:type=”string”
        • Java的数据类型:type=”java.lang.String”
        • 数据库字段的数据类型:
    • 编写映射配置文件

      1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      12
      13
      14
      15
      16
      17
      18
      19
      20
      21
      22
      23
      24
      25
      26
      <?xml version="1.0" encoding="UTF-8"?>
      <!DOCTYPE hibernate-mapping PUBLIC
      "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
      "http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd">
      <hibernate-mapping>
      <!-- 配置类和表结构的映射 -->
      <class name="com.lousen.domain.Customer" table="cst_customer">
      <!-- 配置id,name:JavaBean,column:表结构的字段 -->
      <id name="cust_id" column="cust_id">
      <!-- 主键生成策略 -->
      <generator class="native"></generator>
      </id>
      <!-- 配置其他属性 -->
      <property name="cust_name" column="cust_name"/>
      <property name="cust_user_id" column="cust_user_id"/>
      <property name="cust_create_id" column="cust_create_id"/>
      <property name="cust_source" column="cust_source"/>
      <property name="cust_industry" column="cust_industry"/>
      <property name="cust_level" column="cust_level"/>
      <property name="cust_linkman" column="cust_linkman"/>
      <property name="cust_phone" column="cust_phone"/>
      <property name="cust_mobile" column="cust_mobile"/>
      </class>
      </hibernate-mapping>

Hibernate核心的配置文件

  1. 创建核心的配置文件

    • 指定名称:hibernate.cfg.xml
    • 默认位置:在src目录下
    • 引入约束:在xml配置文件中引入约束(引入的是hibernate3.0的dtd约束,不要引入4的约束)
      1
      2
      3
      <!DOCTYPE hibernate-configuration PUBLIC
      "-//Hibernate/Hibernate Configuration DTD 3.0//EN"
      "http://www.hibernate.org/dtd/hibernate-configuration-3.0.dtd">
  2. 链接本地约束(同上)

  3. 编写核心配置文件

    • 一个Hibernate只需要配置一个核心配置文件

    • 核心配置文件的两种方式

      • 属性文件的形式(不推荐),即properties的配置文件
        • hibernate.properties
          • hibernate.connection.driver_class=com.mysql.jdbc.Driver
        • 缺点
          • 不能加载映射的配置文件,需要手动编写代码去加载
      • XML文件的形式(推荐)
        • hibernate.cfg.xml
          • <property name="hibernate.connection.driver_class" >com.mysql.jdbc.Driver</property>
        • 优点
          • 格式比较清晰
          • 编写有提示
          • 可以在该配置文件中加载映射的配置文件(最主要的)
    • hibernate.cfg.xml的配置方式

      • 必须有的配置
        • 数据库连接信息:
          hibernate.connection.driver_class:连接数据库驱动程序
          hibernate.connection.url:连接数据库URL
          hibernate.connection.username:数据库用户名
          hibernate.connection.password:数据库密码
        • 方言:
          hibernate.dialect:操作数据库方言
      • 可选配置
        • hibernate.show_sql:显示SQL语句
        • hibernate.format_sql:格式化SQL语句
        • hibernate.hbm2ddl.auto:通过映射转成DDL语句
          • create:每次都会创建一个新的表.—测试的时候
          • create-drop:每次都会创建一个新的表,当执行结束之后,将创建的这个表删除(用于测试的时候)
          • update:如果有表,使用原来的表.没有表,创建一个新的表.同时更新表结构(开发中常用)
          • validate:如果有表,使用原来的表.同时校验映射文件与表中字段是否一致如果不一致就会报错(开发完后使用)
      • 加载映射
        • XML方式:<mapping resource="com/lousen/domain/Customer.hbm.xml" />
    • 编写核心配置文件

      1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      12
      13
      14
      15
      16
      17
      18
      19
      20
      21
      22
      23
      24
      25
      26
      27
      28
      29
      30
      31
      32
      33
      34
      <?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>标签,一个数据库对应一个<session-factory> -->
      <session-factory>
      <!-- 必须要配置的参数有5个,4大参数+数据库方言 -->
      <!-- url如果是本地可用/省略,后面的是数据库的名称 -->
      <!-- 可以在hibernate-release-5.0.7.Final/project/etc/hibernate.properties查看所有具体的配置信息 -->
      <property name="hibernate.connection.driver_class">com.mysql.jdbc.Driver</property>
      <property name="hibernate.connection.url">jdbc:mysql:///hibernate_test</property>
      <property name="hibernate.connection.username">lousen</property>
      <property name="hibernate.connection.password">1135497143</property>
      <!-- 数据库的方言 -->
      <property name="hibernate.dialect">org.hibernate.dialect.MySQLDialect</property>
      <!-- 可选配置 -->
      <!-- 显示SQL语句,在控制台显示 -->
      <property name="hibernate.show_sql">true</property>
      <!-- 格式化SQL语句 -->
      <property name="hibernate.format_sql">true</property>
      <!-- 生成数据库的表结构
      update:如果没有表结构,创建表结构。如果存在,不会创建,添加数据
      -->
      <property name="hibernate.hbm2ddl.auto">update</property>
      <!-- 映射配置文件,需要引入映射的配置文件 -->
      <mapping resource="com/itheima/domain/Customer.hbm.xml"/>
      </session-factory>
      </hibernate-configuration>

编写Hibernate入门代码

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
package com.lousen.test;
import org.hibernate.Session;
import org.hibernate.SessionFactory;
import org.hibernate.Transaction;
import org.hibernate.cfg.Configuration;
import org.junit.Test;
import com.lousen.domain.Customer;
/**
* 编写Hibernate的入门代码
* @author LousenJay
*
*/
public class Demo1 {
/*测试保存(添加)用户*/
@Test
public void testSave(){
//1.加载配置文件
Configuration config = new Configuration();
//默认加载src目录下的Hibernate.cfg.xml
config.configure();
//2.创建SessionFactory对象,生成Session对象
SessionFactory factory = config.buildSessionFactory();
//3.创建Session对象
Session session = factory.openSession();
//4.开启事务
Transaction tr = session.beginTransaction();
//5.编写代码
Customer c = new Customer();
c.setCust_name("张三");
c.setCust_level("1");
c.setCust_phone("123456789");
//保存数据,操作对象就相当于操作数据库的表结构
session.save(c);
//6.提交事务
tr.commit();
//7.释放资源
session.close();
factory.close();
}
}

Hibernate常用的接口和类

Configuration类

  • 测试的时候加载Hibernate的配置文件,可以获取SessionFactory对象
  • 如果采用的是properties的配置文件,那么通过Configuration configuration = new Configuration();就可以加载配置文件,但是需要手动加载映射配置文件。
  • 如果采用的XML的配置文件,通过Configuration configuration = new Configuration().configure();加载配置文件。

SessionFactory类(重点)

  • 生成Session对象的工厂类
  • 由Configuration通过加载配置文件创建该对象
  • SessionFactory对象中保存了当前的数据库配置信息和所有映射关系以及预定义的SQL语句。同时,SessionFactory还负责维护Hibernate的二级缓存。
  • 一个SessionFactory实例对应一个数据库,应用从该对象中获得Session实例。
  • SessionFactory是线程安全的,意味着它的一个实例可以被应用的多个线程共享。
  • SessionFactory是重量级的,意味着不能随意创建或销毁它的实例。如果只访问一个数据库,只需要创建一个SessionFactory实例,且在应用初始化的时候完成。
  • SessionFactory需要一个较大的缓存,用来存放预定义的SQL语句及实体的映射信息。另外可以配置一个缓存插件,这个插件被称之为Hibernate的二级缓存,被多线程所共享。
  • 一般应用使用一个SessionFactory,最好是应用启动时就完成初始化。

HibernateUtil工具类

  • 目的:每次创建销毁SessionFactory需要消耗大量资源,所以在一个项目中只使用一个SessionFactory,在应用启动时初始化,应用结束时才销毁。

  • 编写HibernateUtils的工具类

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    package com.lousen.utils;
    /**
    * Hibernate框架的工具类
    * @author LousenJay
    *
    */
    import org.hibernate.Session;
    import org.hibernate.SessionFactory;
    import org.hibernate.cfg.Configuration;
    public class HibernateUtils {
    private static final Configuration CONFIGURATION;
    private static final SessionFactory SESSIONFACTORY;
    //编写静态初始化代码块
    static{
    //加载XML配置文件
    CONFIGURATION = new Configuration().configure();
    //构造工厂
    SESSIONFACTORY = CONFIGURATION.buildSessionFactory();
    }
    /**
    * 从工厂中获取对象
    * @return
    */
    public static Session getSession(){
    return SESSIONFACTORY.openSession();
    }
    }

Session接口

  • 概述

    • Session是在Hibernate中使用最频繁的接口。也被称之为持久化管理器。它提供了和持久化有关的操作,比如添加、修改、删除、加载和查询实体对象
    • Session 是应用程序与数据库之间交互操作的一个单线程对象,是Hibernate 运作的中心
    • Session是线程不安全的
    • 所有持久化对象必须在 session的管理下才可以进行持久化操作
    • Session对象有一个一级缓存,显式执行flush之前,所有的持久化操作的数据都缓存在 session 对象中
    • 持久化类与 Session 关联起来后就具有了持久化的能力
  • 特点

    • 不是线程安全的。应避免多个线程使用同一个Session实例
    • Session是轻量级的,它的创建和销毁不会消耗太多的资源。应为每次客户请求分配独立的Session实例
    • Session有一个缓存,被称之为Hibernate的一级缓存。每个Session实例都有自己的缓存
  • 常用方法

    • save(obj):保存(添加)

      1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      12
      13
      14
      15
      16
      17
      18
      19
      /*测试保存(添加)用户*/
      @Test
      public void testSave(){
      //创建Session对象
      Session session = HibernateUtils.getSession();
      //开启事务
      Transaction tr = session.beginTransaction();
      //编写代码
      Customer c = new Customer();
      c.setCust_name("李四");
      c.setCust_level("1");
      c.setCust_phone("123456789");
      //保存数据,操作对象就相当于操作数据库的表结构
      session.save(c);
      //提交事务
      tr.commit();
      //释放资源
      session.close();
      }
    • delete(obj) :删除

      1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      12
      13
      14
      15
      16
      /*测试delete()方法,先查询再删除*/
      @Test
      public void testDel(){
      //创建Session对象
      Session session = HibernateUtils.getSession();
      //开启事务
      Transaction tr = session.beginTransaction();
      //第一个参数:所查询的JavaBean的class对象,第二个参数:主键值(包装类型)
      Customer c = session.get(Customer.class, 1L);
      //删除该客户
      session.delete(c);
      //提交事务
      tr.commit();
      //释放资源
      session.close();
      }
    • get(Class,id):查询一条数据

      1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      12
      13
      14
      15
      /*测试get()方法,通过主键来查询一条数据*/
      @Test
      public void testGet(){
      //创建Session对象
      Session session = HibernateUtils.getSession();
      //开启事务
      Transaction tr = session.beginTransaction();
      //第一个参数:所查询的JavaBean的class对象,第二个参数:主键值(包装类型)
      Customer c = session.get(Customer.class, 1L);
      System.out.println(c);
      //提交事务
      tr.commit();
      //释放资源
      session.close();
      }
    • update(obj):修改

      1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      12
      13
      14
      15
      16
      17
      18
      19
      /*测试update()方法,先查询再修改*/
      @Test
      public void testUpdate(){
      //创建Session对象
      Session session = HibernateUtils.getSession();
      //开启事务
      Transaction tr = session.beginTransaction();
      //第一个参数:所查询的JavaBean的class对象,第二个参数:主键值(包装类型)
      Customer c = session.get(Customer.class, 3L);
      //设置该客户信息
      c.setCust_industry("电脑");
      //c.setCust_id(2L); //不能修改主键的值,主键的值由策略管理
      //修改该客户信息
      session.update(c);
      //提交事务
      tr.commit();
      //释放资源
      session.close();
      }
    • saveOrUpdate(obj):保存或者修改(有就修改,没有就保存)

      1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      12
      13
      14
      15
      16
      17
      18
      19
      /*测试添加或修改方法,存在的话就是修改,不存在就是保存,操作方法也和修改和保存一样*/
      @Test
      public void testSaveOrUpdate(){
      //创建Session对象
      Session session = HibernateUtils.getSession();
      //开启事务
      Transaction tr = session.beginTransaction();
      //第一个参数:所查询的JavaBean的class对象,第二个参数:主键值(包装类型)
      Customer c = session.get(Customer.class, 3L);
      //设置该客户信息
      c.setCust_industry("手机");
      //c.setCust_id(2L); //不能修改主键的值,主键的值由策略管理
      //修改该客户信息
      session.saveOrUpdate(c);
      //提交事务
      tr.commit();
      //释放资源
      session.close();
      }
    • createQuery():HQL语句的查询的方式

      1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      12
      13
      14
      15
      16
      17
      18
      19
      20
      /*测试查询的方法*/
      @Test
      public void testSelect(){
      //创建Session对象
      Session session = HibernateUtils.getSession();
      //开启事务
      Transaction tr = session.beginTransaction();
      //创建查询的接口,from JavaBean名称
      Query query = session.createQuery("from Customer");
      //查询所有数据:select * from 表
      List<Customer> list = query.list();
      //遍历
      for (Customer customer : list) {
      System.out.println(customer);
      }
      //提交事务
      tr.commit();
      //释放资源
      session.close();
      }

Transaction接口

  • Transaction是事务的接口
  • 常用的方法
    • commit:提交事务
    • rollback:回滚事务
  • 特点
    • Hibernate框架默认情况下事务不自动提交,需要手动提交事务
    • 如果没有开启事务,那么每个Session的操作都相当于一个独立的事务
  • 事务的常规操作
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    /*测试常规事务操作*/
    @Test
    public void testSave3(){
    //创建Session对象,Transaction对象
    Session session = null;
    Transaction tr = null;
    try {
    session = HibernateUtils.getSession();
    //开启事务
    tr = session.beginTransaction();
    //编写代码
    Customer c = new Customer();
    c.setCust_name("王五");
    c.setCust_level("1");
    c.setCust_phone("123456789");
    //保存数据,操作对象就相当于操作数据库的表结构
    session.save(c);
    //提交事务
    tr.commit();
    } catch (Exception e) {
    tr.rollback();
    } finally {
    //释放资源
    session.close();
    }
    }