Hibernate是一个开放源代码的对象关系映射(Object Relation Mapping,简称ORM)框架,同时也是Java领域的一个持久化框架。
1. Hibernate框架
1.1 框架
框架可以看做一个半成品,基于此可提高开发效率、减少代码量、提高程序的健壮性等。
1.2 持久化
狭义的理解,持久化仅指将对象永久保存在数据库中;而广义的理解则包括与数据库相关的增删改查和加载操作。
其中,加载是指根据特定的OID将一个对象从数据库加载到内存中。为了在系统中能够找到所需对象,需要为每一个对象分配唯一的标识号;在关系数据库中称之为主键,而在对象术语中,则叫做对象标识(Object identifier,简称OID)。
1.3 ORM框架
ORM框架的功能是实现对象与关系的映射,其思想是将关系数据库中表的记录映射为对象,程序员可以把对数据库的操作直接转化为对对象的操作。其中,具体映射关系如下图所示:
面向对象概念 | 面向关系概念 |
---|---|
类 | 表 |
对象 | 表的行(记录) |
属性 | 表的列(字段) |
ORM框架采用元数据(描述数据的数据)来描述对象关系映射细节,元数据通常采用XML格式存放在专门的对象关系映射文件中。
其中,ORM框架架构如下图所示:
就目前而言,较为流行的ORM框架主要有Hibernate、myBatis、TopLink、OJB等。其中,Hibernate框架较为成熟、可完成对象的持久化操作、允许开发者采用面向对象的方式来操作关系数据库、可消除针对特定数据库厂商的SQL代码;然而,myBatis框架相比Hibernate更为灵活、运行速度快,但其开发速度慢、不支持纯粹的面向对象操作、需熟悉sql语句并且熟练使用sql语句优化功能。
1.4 插件安装
Hibernate插件可使开发更为简单方便,因此需要在开发前进行安装;在此,选择的安装文件为hibernatetools-Update-4.1.1.Final_2013-12-08_01-06-33-B605.zip。
2. 实现步骤
Hibernate的具体实现步骤示意图如下图所示:
2.1 环境准备
2.2 创建Hibernate配置文件(hibernate.cfg.xml)
Hibernate从其配置文件中读取和数据库连接的有关信息,该文件应位于应用的classpath下。具体示例代码如附录hibernate.cfg.xml文件所示。
2.3 创建持久化Java类
Hibernate被称为低浸入式设计框架,因为其持久化类不需要继承任何父类或实现任何接口,可保证代码不被污染;该持久化类主要有以下要求:
- 提供无参的构造器:使Hibernate需要使用Constructor.newInstance()来实例化持久化类;
- 提供标识属性(identifier property):其映射为数据库表的主键字段,若没有该属性一些功能则不起作用,如Session.saveOrUpdate();
- 为类的持久化类字段声明访问方法(get/set): Hibernate对JavaBeans风格的属性实行持久化;
- 使用非final类: 在运行时生成代理是Hibernate的一个重要功能;如果持久化类没有实现任何接口,Hibnernate使用CGLIB生成代理,但若使用final类,则无法生成CGLIB代理;
- 重写eqauls()和hashCode()方法:如果需要把持久化类的实例放到Set中(当需要进行关联映射时), 则需要重写。
2.4 创建对象关系映射文件(*.hbm.xml)
Hibernate采用XML格式文件来指定对象和关系数据之间的映射,其在运行时 将根据该映射文件来生成所对应的SQL语句。具体示例代码如附录News.hbm.xml文件所示。
2.5 编写代码访问数据库
2.5.1 Configuration类
作用:负责管理Hibernate的配置信息,主要包括Hibernate运行的底层信息(如数据库的URL、用户名、密码、JDBC驱动类、数据库连接池、数据库Dialect等,对应hibernate.cfg.xml文件)和持久化类与数据表的映射关系(对应*.hbm.xml 文件)。
创建Configuration对象:
- 若采用hibernate.properties文件配置,则
Configuration cfg = new Configuration()
; - 若采用hibernate.cfg.xml文件配置,则
Configuration cfg = new Configuration().configure()
,且Configuration对象的configure()方法还支持带参数的访问。
2.5.2 SessionFactory接口
- 针对单个数据库映射关系经过编译后的内存镜像,是线程安全的;
- 构造SessionFactory很耗资源,一般情况下一个应用中只初始化一个SessionFactory对象;
- SessionFactory是生成Session的工厂,该对象一旦构造完毕,即被赋予特定的配置信息;
- Hibernate4新增ServiceRegistry接口,所有基于Hibernate的配置或服务都必须统一向这个ServiceRegistry注册后才能生效。
- 其中,SessionFactory的具体创建步骤如下:
// 1). 创建Configuration对象:对应Hibernate基本配置信息和对象关系映射信息
Configuration configuration = new Configuration().configure();
// Hibernate4.0之前这样创建
// sessionFactory = configuration.buildSessionFactory();
// 2). 创建一个ServiceRegistry对象:Hibernate4.x新增
// Hibernate的任何配置和服务都需要在该对象中注册后才能有效
ServiceRegistry serviceRegistry = new ServiceRegistryBuilder()
.applySettings(configuration.getProperties())
.buildServiceRegistry();
// 3). 创建SessionFactory对象
sessionFactory = configuration.buildSessionFactory(serviceRegistry);
2.5.3 Session接口
Session对象是应用程序与数据库之间交互操作的一个单线程对象,是Hibernate的运作中心,所有持久化对象必须在Session的管理下才可以进行持久化操作;另外,此对象生命周期很短,且具有一个一级缓存,显式执行flush()方法前,所有的持久层操作的数据都缓存在Session对象处,其相当于JDBC中的Connection。
注意:持久化类与Session关联起来后就具有了持久化的能力。其中,Session接口的常用方法主要有:
- 开启事务:beginTransaction();
- 取得持久化对象的方法:get()、load();
- 持久化对象的操作:save()、update()、saveOrUpdate()、delete();
- 管理Session对象:isOpen()、flush()、clear()、evict()、close()等。
2.5.4 Transaction(事务)
其代表一次原子操作,具有数据库事务的概念,即所有持久层操作都应该在事务管理下进行,包括只读操作。其常用方法主要有:
- rollback():撤销事务操作;
- wasCommitted():检查事务是否提交;
- commit():提交事务,即提交相关联的session实例。
附录:示例代码
代码结构:
hibernate.cfg.xml:
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE hibernate-configuration PUBLIC
"-//Hibernate/Hibernate Configuration DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-configuration-3.0.dtd">
<hibernate-configuration>
<session-factory>
<!-- 1. 配置连接数据库的基本信息 -->
<property name="connection.username">root</property>
<property name="connection.password">root</property>
<property name="connection.driver_class">com.mysql.jdbc.Driver</property>
<!-- jdbc:mysql://localhost:3306/hibernate localhost:3306为默认的,可以去除 -->
<property name="connection.url">jdbc:mysql:///hibernate</property>
<property name="connection.username"></property>
<!-- 2. 配置Hibernate的基本信息 -->
<!-- 2.1 配置Hibernate所使用的数据库方言 -->
<!-- 可在hibernate-release-4.2.4.Final\project\etc\hibernate.properties文件中查找 -->
<property name="dialect">hibernate.dialect org.hibernate.dialect.MySQLInnoDBDialect</property>
<!-- 2.2 执行操作时是否在控制台打印SQL语句 -->
<property name="show_sql">true</property>
<!-- 2.3 执行操作时是否是否对SQL进行格式化 -->
<property name="format_sql">true</property>
<!-- 2.4 指定执行操作时自动生成数据表的策略,共有四种取值 -->
<!-- create : 根据*.hbm.xml文件来生成数据表,但每次运行都会删除旧表创建新表,即使数据表内容没有改变;
create-drop : 根据*.hbm.xml文件生成数据表,但SessionFactory对象关闭即删除数据表;
update : 最常用的属性值,也会根据*.hbm.xml文件生成数据表表, 但若*.hbm.xml文件与数据库中对应数据表结构不同,Hiberante将更新数据表结构,但不会删除已有的行和列;
validate : 会与数据库中的表进行比较,若*.hbm.xml文件中的列在数据表中不存在,则抛出异常。
-->
<property name="hbm2ddl.auto">update</property>
<!-- 2.5 指定所关联的对象关系映射文件 -->
<mapping resource="com/qiaobc/hibernate/helloworld/News.hbm.xml"/>
</session-factory>
</hibernate-configuration>
News.java:
package com.qiaobc.hibernate.helloworld;
import java.sql.Date;
public class News {
private Integer id;
private String title;
private String author;
private Date date;
public Integer getId() {
return id;
}
public void setId(Integer id) {
this.id = id;
}
public String getTitle() {
return title;
}
public void setTitle(String title) {
this.title = title;
}
public String getAuthor() {
return author;
}
public void setAuthor(String author) {
this.author = author;
}
public Date getDate() {
return date;
}
public void setDate(Date date) {
this.date = date;
}
public News(Integer id, String title, String author, Date date) {
super();
this.id = id;
this.title = title;
this.author = author;
this.date = date;
}
public News() {
super();
}
@Override
public String toString() {
return "News [id=" + id + ", title=" + title + ", author=" + author + ", date=" + date + "]";
}
}
News.hbm.xml:
<?xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
<!-- Generated 2017-2-19 16:24:21 by Hibernate Tools 3.4.0.CR1 -->
<hibernate-mapping>
<class name="com.qiaobc.hibernate.helloworld.News" table="NEWS">
<id name="id" type="java.lang.Integer">
<column name="ID" />
<!-- 指定主键的生成方式 native:使用数据库本地的方式 -->
<generator class="native" />
</id>
<property name="title" type="java.lang.String">
<column name="TITLE" />
</property>
<property name="author" type="java.lang.String">
<column name="AUTHOR" />
</property>
<property name="date" type="java.sql.Date">
<column name="DATE" />
</property>
</class>
</hibernate-mapping>
HibernateTest.java:
package com.qiaobc.hibernate.helloworld;
import java.sql.Date;
import org.hibernate.Session;
import org.hibernate.SessionFactory;
import org.hibernate.Transaction;
import org.hibernate.cfg.Configuration;
import org.hibernate.service.ServiceRegistry;
import org.hibernate.service.ServiceRegistryBuilder;
import org.junit.Test;
public class HibernateTest {
@Test
public void test() {
// 1. 创建SessionFactory对象
SessionFactory sessionFactory = null;
// 1). 创建Configuration对象:对应Hibernate基本配置信息和对象关系映射信息
Configuration configuration = new Configuration().configure();
// Hibernate4.0之前这样创建
// sessionFactory = configuration.buildSessionFactory();
// 2). 创建一个ServiceRegistry对象:Hibernate4.x新增
// Hibernate的任何配置和服务都需要在该对象中注册后才能有效
ServiceRegistry serviceRegistry = new ServiceRegistryBuilder()
.applySettings(configuration.getProperties())
.buildServiceRegistry();
// 3). 创建SessionFactory对象
sessionFactory = configuration.buildSessionFactory(serviceRegistry);
// 2. 创建Session对象
Session session = sessionFactory.openSession();
// 3. 开启事务
Transaction transaction = session.beginTransaction();
// 4. 执行保存操作
News news = new News(null, "qiaobc", "qiaob", new Date(
new java.util.Date().getTime()));
session.save(news);
// 5. 提交事务
transaction.commit();
// 6. 关闭Session对象
session.close();
// 7. 关闭SessionFactory对象
sessionFactory.close();
}
}