hibernate是一个ORM(对象关系映射)框架,ORM简单的说就是我们的实体对象与数据表的映射,由实体对象映射出数据表。hibernate是非常优秀的dao层的框架之一。使用hibernate框架,可以大大提高我们的开发效率,摆脱繁琐的jdbc原始操作,甚至都不用写sql语句了,这是我们程序员最希望的啦。
搭建hibernate工程非常简单大体可以分为以下几个步骤。
第一步:导包(hibernate的核心包以及数据库驱动包)
第二步:构建实体类(修饰实体成员变量时,最好使用包装类型来修饰)
第三步:书写orm元数据文件
第四步:书写hibernate主配置文件
第五步:书写测试代码
以上就是较为常规的开发步骤。接下来笔者以实际案例来驱动。----crm系统中客户管理模块
第一步:在我们的web工程里,导入hibernate核心包以及数据库驱动包(如下图)
第二步:构建实体类Customer.java(因为面向对象开发)
public class Customer {
/*
* CREATE TABLE `cst_customer` (
`cust_id` BIGINT(32) NOT NULL AUTO_INCREMENT COMMENT '客户编号(主键)',
`cust_name` VARCHAR(32) NOT NULL COMMENT '客户名称(公司名称)',
`cust_source` VARCHAR(32) DEFAULT NULL COMMENT '客户信息来源',
`cust_industry` VARCHAR(32) DEFAULT NULL COMMENT '客户所属行业',
`cust_level` VARCHAR(32) DEFAULT NULL COMMENT '客户级别',
`cust_linkman` VARCHAR(64) DEFAULT NULL COMMENT '联系人',
`cust_phone` VARCHAR(64) DEFAULT NULL COMMENT '固定电话',
`cust_mobile` VARCHAR(16) DEFAULT NULL COMMENT '移动电话',
PRIMARY KEY (`cust_id`)
) ENGINE=INNODB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8;
*/
private Long cust_id;
private String cust_name;
private String cust_source;
private String cust_industry;
private String cust_level;
private String cust_linkman;
private String cust_phone;
private String cust_mobile;
public Long getCust_id() {
return cust_id;
}
public void setCust_id(Long cust_id) {
this.cust_id = cust_id;
}
public String getCust_name() {
return cust_name;
}
public void setCust_name(String cust_name) {
this.cust_name = cust_name;
}
public String getCust_source() {
return cust_source;
}
public void setCust_source(String cust_source) {
this.cust_source = cust_source;
}
public String getCust_industry() {
return cust_industry;
}
public void setCust_industry(String cust_industry) {
this.cust_industry = cust_industry;
}
public String getCust_level() {
return cust_level;
}
public void setCust_level(String cust_level) {
this.cust_level = cust_level;
}
public String getCust_linkman() {
return cust_linkman;
}
public void setCust_linkman(String cust_linkman) {
this.cust_linkman = cust_linkman;
}
public String getCust_phone() {
return cust_phone;
}
public void setCust_phone(String cust_phone) {
this.cust_phone = cust_phone;
}
public String getCust_mobile() {
return cust_mobile;
}
public void setCust_mobile(String cust_mobile) {
this.cust_mobile = cust_mobile;
}
@Override
public String toString() {
return "Customer [cust_id=" + cust_id + ", cust_name=" + cust_name + "]";
}
}
第三步:书写orm元数据映射文件--Customer.hbm.xml
将改文件与上一步骤的实体类放在一个包下。代码如下
<!DOCTYPE hibernate-mapping PUBLIC
"-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd">
<hibernate-mapping package="cn.yzj.hibernate.domain">
<class name="Customer" table="cst_customer">
<id name="cust_id" column="cust_id">
<generator class="native"></generator>
</id>
<property name="cust_name" column="cust_name" >
<!-- <column name="cust_name" sql-type="varchar" ></column> -->
</property>
<property name="cust_source" column="cust_source" ></property>
<property name="cust_industry" column="cust_industry" ></property>
<property name="cust_level" column="cust_level" ></property>
<property name="cust_linkman" column="cust_linkman" ></property>
<property name="cust_phone" column="cust_phone" ></property>
<property name="cust_mobile" column="cust_mobile" ></property>
</class>
</hibernate-mapping>
这里笔者详细解释一下此映射文件内的具体内容。
<class>里name指代是我们这个实体类的类名,table是指代数据库表里的某一张表名,整一句就是要将这个实体类与数据表形成映射关系,这也就是orm的体现之处。
里面的<id>属性是指将要生成的表里的主键。<id>里的name属性是实体类里成员变量的主键名称,column是指数据库表对应主键的列名。这里的column可以不写,不写时,将默认将name的值一样。
再者,<generator class="native"></generator>表示主键生成策略,不同生成策略有不同效果。分别如下:
identity : 主键自增.由数据库来维护主键值.录入时不需要指定主键.
sequence: Oracle中的主键生成策略.
increment(了解): 主键自增.由hibernate来维护.每次插入前会先查询表中id最大值.+1作为新主键值.
hilo(了解): 高低位算法.主键自增.由hibernate来维护.开发时不使用.
native:hilo+sequence+identity 自动三选一策略.
uuid: 产生随机字符串作为主键. 主键类型必须为string 类型.
assigned:自然主键生成策略. hibernate不会管理主键值.由开发人员自己录入.
这里笔者推荐选择native。省事方便!
在orm映射中,我们已经搞定了主键的映射解释,接下来解释普通属性的解释。
除主键之外的字段,我们用<property>来指明映射。同样的,name指的是实体类里成员变量名称,column指的是将要映射成的列名名字
这样,我们就完成了一个orm元数据映射了。
第四步:书写hibernate主配置文件---hibernate.cfg.cml。
<?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>
<!--
#hibernate.dialect org.hibernate.dialect.MySQLDialect
#hibernate.dialect org.hibernate.dialect.MySQLInnoDBDialect
#hibernate.dialect org.hibernate.dialect.MySQLMyISAMDialect
#hibernate.connection.driver_class com.mysql.jdbc.Driver
#hibernate.connection.url jdbc:mysql:///test
#hibernate.connection.username gavin
#hibernate.connection.password
-->
<!-- 数据库驱动 -->
<property name="hibernate.connection.driver_class">com.mysql.jdbc.Driver</property>
<!-- 数据库url -->
<property name="hibernate.connection.url">jdbc:mysql:///hibernate_32</property>
<!-- 数据库连接用户名 -->
<property name="hibernate.connection.username">root</property>
<!-- 数据库连接密码 -->
<property name="hibernate.connection.password">1234</property>
<!-- 数据库方言
不同的数据库中,sql语法略有区别. 指定方言可以让hibernate框架在生成sql语句时.针对数据库的方言生成.
sql99标准: DDL 定义语言 库表的增删改查
DCL 控制语言 事务 权限
DML 操纵语言 增删改查
注意: MYSQL在选择方言时,请选择最短的方言.
-->
<property name="hibernate.dialect">org.hibernate.dialect.MySQLDialect</property>
<!-- #hibernate.show_sql true
#hibernate.format_sql true
-->
<!-- 将hibernate生成的sql语句打印到控制台 -->
<property name="hibernate.show_sql">true</property>
<!-- 将hibernate生成的sql语句格式化(语法缩进) -->
<property name="hibernate.format_sql">true</property>
<!--
## auto schema export 自动导出表结构. 自动建表
#hibernate.hbm2ddl.auto create 自动建表.每次框架运行都会创建新的表.以前表将会被覆盖,表数据会丢失.(开发环境中测试使用)
#hibernate.hbm2ddl.auto create-drop 自动建表.每次框架运行结束都会将所有表删除.(开发环境中测试使用)
#hibernate.hbm2ddl.auto update(推荐使用) 自动生成表.如果已经存在不会再生成.如果表有变动.自动更新表(不会删除任何数据).
#hibernate.hbm2ddl.auto validate 校验.不自动生成表.每次启动会校验数据库中表是否正确.校验失败.
-->
<property name="hibernate.hbm2ddl.auto">update</property>
<!-- 引入orm元数据
路径书写: 填写src下的路径
-->
<mapping resource="cn/itheima/domain/Customer.hbm.xml" />
</session-factory>
</hibernate-configuration>
我们来仔细看一下里面的配置内容。这里我们4个必须配置+1个方言,两个可选配置,一个orm元数据映射文件引入。
4个必选配置就是我们之前jdbc常见的四大参数--驱动、数据库url、数据库账号以及密码。
一个方言。每门数据库都有自己的方言,方言就是自己特有的,例如mysql有独有的limit 但是oracle没有,这里limit就是方言之一。在选择mysql方言的时候,hibernate官方提供了三种选择,但是我们需要选择最短那个,就是hibernate.dialect org.hibernate.dialect.MySQLDialect
两个可选配置是SQL语句的打印以及格式化,这里可有可无,看我们自己的需求
最后一个是orm元数据映射文件的引入。
到这里,hibernate的环境已经搭建成功,接下来就写代码来测试一下。
写一个测试方法
@Test
public void hibernate() {
Configuration conf = new Configuration().configure();
SessionFactory sessionFactory = conf.buildSessionFactory();
Session session = sessionFactory.openSession();
Transaction tx = session.beginTransaction();
tx.begin();
// ...................................................
Customer customer = new Customer();
customer.setCust_name("唯品会");
session.save(customer);
// ..................................................
tx.commit();
session.close();
sessionFactory.close();
}
控制台结果:
数据库表的数据结果:
现在我们来仔细看看hibernate框架给我们的提供那套api的用法。首先,我们需要读取hibernate主配置文件,
读取之前我们需要一个用来的读取的工具--Configuration conf = new Configuration(),来得到一个接下来可以读取配置的Configuration 对象conf,注意这里,它还没读取,是空的。接下来,我们就要读取了。hibernate提供了好几个重载方法给我们选择,如下
为什么我们选择无参那个呢?我们来看一下它的源码就可以非常清楚了
看到源码这里,有没有恍然大悟,为什么我们的主配置文件的名称之所以名为hibernate.cfg.xml就是这个原因,hibernate框架会自动帮我查找无需我们手动写文件路径。源码的本志其实也是调用了这个方法。所以我们以后只要保证主配置文件的名称没错,就可以直接调用无参那个方法了来读取配置文件啦。
在hibernate框架里,操作数据库的核心是由session来控制的。所以我们需要在sessionFactory中得到一个session。
所以
SessionFactory sessionFactory = conf.buildSessionFactory();
Session session = sessionFactory.openSession();
这样我们就得到了一个session了。因为这里没有整合spring,所以事务管理这部分,需要我们自己手动控制。首先开启事务
Transaction tx = session.beginTransaction();
tx.begin();
开启事务之后,就可以进行数据库操作了。
Customer customer = new Customer();
customer.setCust_name("唯品会");
session.save(customer);
然后提交事务并关闭资源
tx.commit();
session.close();
sessionFactory.close();
至此,整个程序就结束了。