【1】什么是Hibernate
一个框架,一个Java领域的持久化框架,一个ORM框架。
① 什么是对象的持久化
狭义的理解,“持久化”仅仅指把对象永久保存到数据库中。
广义的理解,“持久化”包括和数据库相关的各种操作:
保存:把对象永久保存到数据库中。
更新:更新数据库中对象(记录)的状态。
删除:从数据库中删除一个对象。
查询:根据特定的查询条件,把符合查询条件的一个或多个对象从数据库加载到内存中。
加载:根据特定的OID,把一个对象从数据库加载到内存中。
什么是OID?
为了在系统中能够找到所需对象,需要为每一个对象分配一个唯一的标识号。在关系数据库中称之为主键,而在对象术语中,则叫做对象标识(Object identifier-OID)。
② 什么是ORM
ORM(Object Relation Mapping): 对象关系映射。
ORM 框架主要解决对象-关系的映射:
ORM的思想:将关系数据库中表中的记录映射成为对象,以对象的形式展现,程序员可以把对数据库的操作转化为对对象的操作。
ORM 采用元数据来描述对象-关系映射细节, 元数据通常采用 XML 格式, 并且存放在专门的对象-关系映射文件中。
其他ORM框架还有MyBatis、TopLink(EclipseLink)以及OJB等。
Hibernate:
非常优秀、成熟的 ORM 框架。
完成对象的持久化操作
Hibernate 允许开发者采用面向对象的方式来操作关系数据库。
消除那些针对特定数据库厂商的 SQL 代码
MyBatis:
相比 Hibernate 灵活高,运行速度快;
开发速度慢,不支持纯粹的面向对象操作,需熟悉sql语句,并且熟练使用sql语句优化功能 。
【2】Hibernate环境配置
① eclipse下hibernatetools安装
为了方便使用Hibernate,你可能需要在eclipse里面安装一个hibernatetools。
如下图所示,你可以在Eclipse Marketplace中安装JBoss Tools 时,单独选择hibernate tools 进行安装:
安装好后效果如下图:
② pom文件
这里Spring和Hibernate版本用的都是5.X。
完整pom文件如下所示:
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd"> <modelVersion>4.0.0</modelVersion> <groupId>com.jane</groupId> <artifactId>hibernate</artifactId> <packaging>war</packaging> <version>0.0.1-SNAPSHOT</version> <name>hibernate Maven Webapp</name> <url>http://maven.apache.org</url> <properties> <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding> <java.version>1.8</java.version> <spring.version>5.0.2.RELEASE</spring.version> <hibernate.version>5.2.10.Final</hibernate.version> <hibernate.validator.version>5.4.0.Final</hibernate.validator.version> <junit.version>4.11</junit.version> <servlet.version>3.1.0</servlet.version> <servlet.api.version>2.5</servlet.api.version> <jsp-api.version>2.3.1</jsp-api.version> <jstl.version>1.2</jstl.version> <mysql.driver.version>5.1.40</mysql.driver.version> <alibaba.druid.version>1.1.10</alibaba.druid.version> <alibaba.fastjson.version>1.2.33</alibaba.fastjson.version> <commons-fileupload.version>1.3.1</commons-fileupload.version> <slf4j.version>1.7.21</slf4j.version> <slf4j.api.version>1.7.25</slf4j.api.version> <log4j.version>1.2.17</log4j.version> <junit.version>4.11</junit.version> <jedis.version>2.9.0</jedis.version> <aspectj.version>1.7.1</aspectj.version> </properties> <dependencies> <!-- spring 包 --> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-jms</artifactId> <version>${spring.version}</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-core</artifactId> <version>${spring.version}</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-tx</artifactId> <version>${spring.version}</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-beans</artifactId> <version>${spring.version}</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-context</artifactId> <version>${spring.version}</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-context-support</artifactId> <version>${spring.version}</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-web</artifactId> <version>${spring.version}</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-webmvc</artifactId> <version>${spring.version}</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-aop</artifactId> <version>${spring.version}</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-aspects</artifactId> <version>${spring.version}</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-jdbc</artifactId> <version>${spring.version}</version> </dependency> <dependency> <groupId>org.aspectj</groupId> <artifactId>aspectjweaver</artifactId> <version>${aspectj.version}</version> </dependency> <!-- Hibernate包 --> <dependency> <groupId>org.hibernate</groupId> <artifactId>hibernate-core</artifactId> <version>${hibernate.version}</version> </dependency> <dependency> <groupId>org.hibernate</groupId> <artifactId>hibernate-validator</artifactId> <version>${hibernate.validator.version}</version> </dependency> <dependency> <groupId>org.hibernate</groupId> <artifactId>hibernate-entitymanager</artifactId> <version>${hibernate.version}</version> </dependency> <!--servlet 包 --> <dependency> <groupId>javax.servlet</groupId> <artifactId>javax.servlet-api</artifactId> <version>${servlet.version}</version> </dependency> <dependency> <groupId>javax.servlet</groupId> <artifactId>servlet-api</artifactId> <version>${servlet.api.version}</version> </dependency> <dependency> <groupId>javax.servlet.jsp</groupId> <artifactId>javax.servlet.jsp-api</artifactId> <version>${jsp-api.version}</version> </dependency> <dependency> <groupId>javax.servlet</groupId> <artifactId>jstl</artifactId> <version>${jstl.version}</version> </dependency> <!-- mysql --> <dependency> <groupId>mysql</groupId> <artifactId>mysql-connector-java</artifactId> <version>${mysql.driver.version}</version> </dependency> <!-- mysql 阿里连接池 druid --> <dependency> <groupId>com.alibaba</groupId> <artifactId>druid</artifactId> <version>${alibaba.druid.version}</version> </dependency> <!-- json 包 --> <dependency> <groupId>com.alibaba</groupId> <artifactId>fastjson</artifactId> <version>${alibaba.fastjson.version}</version> </dependency> <!-- 文件上传包 --> <dependency> <groupId>commons-fileupload</groupId> <artifactId>commons-fileupload</artifactId> <version>${commons-fileupload.version}</version> </dependency> <!-- 日志包 --> <dependency> <groupId>log4j</groupId> <artifactId>log4j</artifactId> <version>${log4j.version}</version> </dependency> <dependency> <groupId>org.slf4j</groupId> <artifactId>slf4j-api</artifactId> <version>${slf4j.api.version}</version> </dependency> <dependency> <groupId>org.slf4j</groupId> <artifactId>slf4j-log4j12</artifactId> <version>${slf4j.version}</version> </dependency> <!-- junit 测试 --> <dependency> <groupId>junit</groupId> <artifactId>junit</artifactId> <version>${junit.version}</version> <scope>provided</scope> </dependency> </dependencies> <build> <finalName>hibernate</finalName> </build> </project>
【3】Hibernate的HelloWorld
Hibernate开发步骤如下:
- 环境配置;
- 配置hibernate.cfg.xml;
- 创建persistent objects;
- 配置*.hbm.xml–对象关系映射文件;
- 通过 Hibernate API 编写访问数据库的代码。
hibernate.cfg.xml配置如下:
<?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> <!-- 配置连接数据库的基本信息 --> <property name="connection.username">root</property> <property name="connection.password">123456</property> <property name="connection.driver_class">com.mysql.jdbc.Driver</property> <property name="connection.url">jdbc:mysql:///test</property> <!-- 配置 hibernate 的基本信息 --> <!-- hibernate 所使用的数据库方言-- 需要注意MySQL5InnoDBDialect 不等于MySQLInnoDBDialect --> <property name="dialect">org.hibernate.dialect.MySQL5InnoDBDialect</property> <!-- 执行操作时是否在控制台打印 SQL --> <property name="show_sql">true</property> <!-- 是否对 SQL 进行格式化 --> <property name="format_sql">true</property> <!-- 指定自动生成数据表的策略--update表示不存在则创建, 存在则不会再次创建表进行覆盖 --> <property name="hbm2ddl.auto">update</property> <!-- 指定关联的 .hbm.xml 文件 --> <mapping resource="com/jane/model/News.hbm.xml"/> </session-factory> </hibernate-configuration>
实例类如下:
Hibernate 不要求持久化类继承任何父类或实现接口,这可以保证代码不被污染。这就是Hibernate被称为低侵入式设计的原因。
public class News { private Integer id; //field private String title; private String author; private String describle; private Date date; //使用 title + "," + content 可以来描述当前的 News 记录. //即 title + "," + content 可以作为 News 的 describle 属性值 private String content; private Blob picture; public Blob getPicture() { return picture; } public void setPicture(Blob picture) { this.picture = picture; } public String getContent() { return content; } public void setContent(String content) { this.content = content; } public Integer getId() { //property 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 String getDescrible() { return describle; } public void setDescrible(String describle) { this.describle = describle; } public News(String title, String author, String describle, Date date, String content, Blob picture) { super(); this.title = title; this.author = author; this.describle = describle; this.date = date; this.content = content; this.picture = picture; } public News() { // TODO Auto-generated constructor stub } @Override public String toString() { return "News [id=" + id + ", title=" + title + ", author=" + author + ", describle=" + describle + ", date=" + date + ", content=" + content + ", picture=" + picture + "]"; } }
Note :
提供一个无参的构造器:使Hibernate可以使用Constructor.newInstance() 来实例化持久化类;
提供一个标识属性(identifier property): 通常映射为数据库表的主键字段. 如果没有该属性,一些功能将不起作用,如:Session.saveOrUpdate();
为类的持久化类字段声明访问方法(get/set): Hibernate对JavaBeans 风格的属性实行持久化;
使用非 final 类: 在运行时生成代理是 Hibernate 的一个重要的功能. 如果持久化类没有实现任何接口, Hibnernate 使用 CGLIB 生成代理. 如果使用的是 final 类, 则无法生成 CGLIB 代理;
重写 eqauls 和 hashCode 方法: 如果需要把持久化类的实例放到 Set 中(当需要进行关联映射时), 则应该重写这两个方法。
根据实例类使用HibernateTools插件创建的News.hbm.xml如下:
Hibernate 采用 XML 格式的文件来指定对象和关系数据之间的映射. 在运行时 Hibernate 将根据这个映射文件来生成各种 SQL 语句。
<?xml version="1.0"?> <!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN" "http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd"> <hibernate-mapping> <class name="com.jane.model.News" table="NEWS"> <id name="id" type="java.lang.Integer"> <column name="ID" /> <!-- 指定主键的生成方式, native: 使用数据库本地方式 --> <generator class="native" /> <!-- <generator class="assigned" /> --> </id> <property name="title" type="java.lang.String"> <column name="TITLE" default="null" /> </property> <property name="author" type="java.lang.String"> <column name="AUTHOR" default="null" /> </property> <property name="describle" type="java.lang.String"> <column name="DESCRIBLE" default="null" /> </property> <property name="date" type="java.util.Date"> <column name="DATE" default="null" /> </property> <property name="content" type="java.lang.String"> <column name="CONTENT" default="null" /> </property> <property name="picture" type="java.sql.Blob"> <column name="PICTURE" default="null" /> </property> </class> </hibernate-mapping>
这里需要注意ID-generator,其选项如下:
测试类如下:
import java.sql.Date; import org.hibernate.Session; import org.hibernate.SessionFactory; import org.hibernate.Transaction; import org.hibernate.boot.MetadataSources; import org.hibernate.boot.registry.StandardServiceRegistry; import org.hibernate.boot.registry.StandardServiceRegistryBuilder; import org.hibernate.cfg.Configuration; import org.junit.Test; import com.jane.model.News; public class HibernateTest { @Test public void test() { System.out.println("test..."); //1. 创建一个 SessionFactory 对象 SessionFactory sessionFactory = null; StandardServiceRegistry ssr = new StandardServiceRegistryBuilder().configure().build(); sessionFactory = new MetadataSources(ssr).buildMetadata().buildSessionFactory(); //5. 创建一个 Session 对象 Session session = sessionFactory.openSession(); //6. 开启事务 Transaction transaction = session.beginTransaction(); //7. 执行保存操作 News news =new News("hibernate", "jane", "orm", new Date(new java.util.Date().getTime()), null, null); System.out.println(news); session.save(news); //8. 提交事务 transaction.commit(); //9. 关闭 Session session.close(); //10. 关闭 SessionFactory 对象 sessionFactory.close(); } }
这里需要注意一个问题,你获取SessionFactory的方式和你的Hibernate有关!
//创建 Configuration 对象: 对应 hibernate 的基本配置信息和 对象关系映射信息 Configuration configuration = new Configuration().configure("hibernate.cfg.xml"); //4.0 之前这样创建 sessionFactory = configuration.buildSessionFactory(); //4.X你可能这样创建 // 创建一个 ServiceRegistry 对象: hibernate 4.x 新添加的对象 //hibernate 的任何配置和服务都需要在该对象中注册后才能有效. ServiceRegistry serviceRegistry = new ServiceRegistryBuilder().applySettings(configuration.getProperties()) .buildServiceRegistry(); //或者这样创建一个 ServiceRegistry 对象 StandardServiceRegistry serviceRegistry = new StandardServiceRegistryBuilder() .applySettings(configuration.getProperties()) .build(); //创建一个 SessionFactory 对象 sessionFactory = configuration.buildSessionFactory(serviceRegistry);
但是5.X中上面示例都过时,请使用如下方式获取SessionFactory:
public class HibernateUtils { private static SessionFactory sessionFactory; private static SessionFactory buildSessionFactory(){ StandardServiceRegistry ssr = new StandardServiceRegistryBuilder().configure().build(); sessionFactory = new MetadataSources(ssr).buildMetadata().buildSessionFactory(); return sessionFactory; } public static SessionFactory getSessionFactory(){ return (sessionFactory==null ? buildSessionFactory() : sessionFactory); } public static Session openSession(){ return getSessionFactory().openSession(); } }
否则你很可能遇到一个org.hibernate.MappingException:Unknown entity异常!