那什么是Hibernate?
Hibernate是一个开放源代码的对象关系映射框架,它对JDBC进行了非常轻量级的对象封装,它将POJO与数据库表建立映射关系,是一个全自动的ORM框架,Hibernate可以自动生成SQL语句,自动执行,使得Java程序员可以随心所欲地使用对象编程思维来操纵数据库,Hibernate可以应用在任何使用JDBC的场合,既可以在Java的客户端程序使用,也可以在Servlet/JSP的Web应用中使用。
顺带介绍一下什么是ORM。
ORM:对象关系映射,是一种程序技术,简单来说就是把我们程序中的实体类和数据库表建立起来对应关系。
那为什么要有对象关系映射呢?
假设一下,当你开发一个应用程序的时候,你可能会写不少数据访问层的代码,用来从数据库保存、删除、读取对象信息等等。你在DAO中写了很多的方法来读取对象数据,改变状态对象等等任务,而这些代码很多都是重复的。
对象映射关系赋予了程序一种强大的能力,让开发者仅仅掌握面向对象的思维来操作数据库,即在关系型数据库和业务实体对象之间作一个映射,这样,我们在具体的操作业务对象的时候,就不需要再去和复杂的SQL语句打交道,只要像平时操作对象一样操作它就可以了。
HelloWorld
关于Hibernate的一些基本知识了解完之后,我们来编写一个Hibernate的入门项目。
首先我们得编写hibernate的配置文件,在src目录下新建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> <!-- 配置连接数据库的基本信息 --> <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的基本信息 --> <property name="dialect">org.hibernate.dialect.MySQL5InnoDBDialect</property> <!-- 执行操作时是否在控制台打印sql --> <property name="show_sql">true</property> <!-- 是否对sql进行格式化 --> <property name="format_sql">true</property> <!-- 指定自动生成数据表的策略 --> <property name="hbm2ddl.auto">update</property> <!-- 指定关联的.hbm.xml文件 --> <mapping resource="com/itcast/hibernate/helloworld/Account.hbm.xml" /> </session-factory> </hibernate-configuration>
这里面就是配置一些相关的信息。
然后我们创建一个Bean类Account:
package com.itcast.hibernate.helloworld; public class Account { private Integer id; private String name; private double money; public Account() { } public Account(String name, double money) { this.name = name; this.money = money; } public Integer getId() { return id; } public void setId(Integer id) { this.id = id; } public String getName() { return name; } public void setName(String name) { this.name = name; } public double getMoney() { return money; } public void setMoney(double money) { this.money = money; } @Override public String toString() { return "Account [id=" + id + ", name=" + name + ", money=" + money + "]"; } }
接下来我们需要创建对象关系映射的配置文件,在与该类同级的目录下新建Account.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"> <hibernate-mapping> <!-- 使Account类对应数据表ACCOUNT --> <class name="com.itcast.hibernate.helloworld.Account" table="ACCOUNT"> <!-- id标签中的name为类中的属性名;colum标签中的name为数据表中的列名 --> <id name="id" type="java.lang.Integer"> <column name="ID" /> <!-- 指定主键的生成方式 native:使用数据库本地的方式--> <generator /> </id> <!-- property表示非id的其它列 --> <property name="name" type="java.lang.String"> <column name="NAME" /> </property> <property name="money" type="double"> <column name="MONEY" /> </property> </class> </hibernate-mapping>
简单地配置一下,该配置文件目的是使类和表进行一个映射。
然后我们编写一下测试代码:
// 1、创建一个SessionFactory对象 SessionFactory sessionFactory = null; // 1)、创建Configuration对象:对应hibernate的基本配置信息和对象关系映射信息 // 默认关联的是hibernate.hbm.xml文件,如果你的配置文件名是hibernate.hbm.xml,使用无参构造即可 Configuration configuration = new Configuration().configure(); sessionFactory = configuration.buildSessionFactory(); // 2、创建一个Session对象 Session session = sessionFactory.openSession(); // 3、开启事务 Transaction transaction = session.beginTransaction(); // 4、执行保存操作 Account account = new Account("张三",1000); session.save(account); // 5、提交事务 transaction.commit(); // 6、关闭Session session.close(); // 7、关闭SessionFactory sessionFactory.close();
然后运行测试代码,hibernate就会把数据保存到数据库,如果你没有这张表,hibernate会自动帮我们创建好表然后插入数据。
因为hibernate版本的问题,所以这里面会有很多的坑,具体报了什么错大家可以自己百度解决。我这里的hibernate版本是5.X的。
+----+------+-------+ | ID | NAME | MONEY | +----+------+-------+ | 1 | 张三 | 1000 | +----+------+-------+ 1 row in set (0.00 sec)
那么在创建持久化类的时候需要注意下面几个问题:
- 必须提供一个无参的构造器
- 因为Hibernate需要使用Constructor.newInstance()来实例化持久化类
- 提供一个标识属性
- 通常映射为数据表的主键字段,如果没有该属性,一些功能将不起作用
- 为类的持久化类字段声明访问方法
- 使用非final类
- 在运行时生成代理是Hibernate的一个重要的功能,如果持久化类没有实现任何接口,Hibernate将使用CGLIB生成代理,如果使用的是final类,则无法生成CGLIB代理
- 重写equals和hashCode方法
- 如果需要把持久化类的实例放到Set中(当需要进行关联映射时),则应该重写这两个方法
入门案例我们就编写完了, 相信两个配置文件的配置信息大家都能明白,注释也写得很清楚,那么关于测试代码的一些类API,这里分别介绍一下。
- Configuration:
- Configuration类负责管理Hibernate的配置信息,包括如下内容:
- Hibernate运行的底层信息:数据库的URL、用户名、密码、JDBC驱动类,数据库Dialect、数据库连接池等(对应hibernate.cfg.xml文件)
- 创建Configuration的两种方式:
- 属性文件(hibernate.properties):
- Configuration cfg = new Configuration();
- Xml文件(hibernate.cfg.xml)
- Configuration cfg = new Configuration().configure();
- Configuration的configure方法还支持带参访问:
- File file = new File("文件名");
- Configuration cfg = new Configuration().configure(file);
- SessionFactory:
- 针对单个数据库映射关系经过编译后的内存镜像,是线程安全的
- SessionFactory对象一旦构造完毕,即被赋予特定的配置信息
- SessionFactory是生产Session的工厂
- 构造SessionFactory很消耗资源,一般情况下一个应用中只初始化一个SessionFactory对象
- Session:
- Session是应用程序与数据库之间交互操作的一个而单线程对象,是Hibernate运作的中心,所有持久化对象必须在session的管理下才可以进行持久化操作。此对象的生命周期很短。Session对象有一个一级缓存,显式执行flush之前,所有的持久化层操作的数据都缓存在session对象处,相当于JDBC中的Connection
- Transaction:
- 事务,学过数据库的应该都明白,不重复说
更新
上面说到了插入数据,那么这里说一下如何使用hibernate完成更新操作。
- session.save(obj); 【保存一个对象】
- session.update(obj); 【更新一个对象】
- session.saveOrUpdate(obj); 【保存或者更新的方法】
- 没有设置主键,执行保存;
- 有设置主键,执行更新操作;
- 如果设置主键不存在报错!
SessionFactory sessionFactory = null; Configuration configuration = new Configuration().configure(); sessionFactory = configuration.buildSessionFactory(); Session session = sessionFactory.openSession(); Transaction transaction = session.beginTransaction(); Account account = session.get(Account.class, 1); account.setMoney(2000); session.update(account); transaction.commit(); session.close(); sessionFactory.close();
运行然后查询数据库:
+----+------+-------+ | ID | NAME | MONEY | +----+------+-------+ | 1 | 张三 | 2000 | +----+------+-------+ 1 row in set (0.00 sec)
查询
接下来是查询。
SessionFactory sessionFactory = null; Configuration configuration = new Configuration().configure(); sessionFactory = configuration.buildSessionFactory(); Session session = sessionFactory.openSession(); Transaction transaction = session.beginTransaction(); Account account = session.load(Account.class, 1); System.out.println(account); transaction.commit(); session.close(); sessionFactory.close();
运行结果:
删除
最后是删除。
SessionFactory sessionFactory = null; Configuration configuration = new Configuration().configure(); sessionFactory = configuration.buildSessionFactory(); Session session = sessionFactory.openSession(); Transaction transaction = session.beginTransaction(); Account account = session.get(Account.class, 1); session.delete(account); transaction.commit(); session.close(); sessionFactory.close();
运行结果:
mysql> select * from account; Empty set (0.00 sec)
总结
关于增删改查的操作是非常简单的,只要会一个,其它的就都能会,因为道理是一样的。