以前大多数时间都是做.NET,对JAVA的开源框架这块就简单的了解了下,没怎么认真的学习过,从今天开始我又将参与一个J2EE的项目,因公司的同事几乎全是.NET,今天就和同事一起学习Hibernate这玩意,简单记录些笔记于此,以便以后方便查看,也给初学Hibernate的朋友一点参考.文章中我会将理论和实例结合.

一、初识Hibernate
       Hibernate是........这个我就不多说了,网上到处都可以搜索到.
  
 二、配置Hibernate开发环境
       在开发Hibernate应用的时候需要预先把相应的JAR包配置到CLASS_PATH,我使用的是MyEclipse作为开发工具,主要的JAR大致如下:
       ---antlr.jar
       ---cglib.jar
       ---asm.jar
       ---asm-attrs.jar
       ---commons-collections.jar
       ---commons-logging.jar
       ---ehcache.jar
       ---hibernate3.jar
       ---jta.jar
       ---dom4j.jar
       ---log4j.jar
       这些包从何来,它们都可以从Hibernate网站下载的Hibernate发布包里找到,在lib目录下。
       另外就是还需要一个连接数据库的驱动,我使用的是Oracle数据库。

三、开发Hibernate程序的步骤
      开发Hibernate程序通常分为下面这几步:
      1、 建立JavaBean--> 每个JavaBean必须提供一个无参的构造函数,并为private属性提供get/set访问方法。如下:
 1 package day.one;
 2 
 3 public class Person {
 4     private int id;
 5     private String name;
 6     private int age;
 7     private String phone;
 8     
 9     public int getAge() {
10         return age;
11     }
12 
13     public void setAge(int age) {
14         this.age = age;
15     }
16 
17     public int getId() {
18         return id;
19     }
20 
21     public void setId(int id) {
22         this.id = id;
23     }
24 
25     public String getName() {
26         return name;
27     }
28 
29     public void setName(String name) {
30         this.name = name;
31     }
32 
33     public String getPhone() {
34         return phone;
35     }
36 
37     public void setPhone(String phone) {
38         this.phone = phone;
39     }
40 
41     public Person() {    
42     }
43 
44     public Person(String name, int age, String phone) {
45         super();
46         this.name = name;
47         this.age = age;
48         this.phone = phone;
49     }
50 }

        2、编写映射文件(*.hbm.xml)-->这一步在Hibernate开发中最为重要,工作量是很大的。详细配置格式可查看Hibernate包下的src/org/hibernate/hibernate-mapping-3.0.dtd,下面代码块便是上面Person的配置文件:

 1 <?xml version="1.0"?>
 2 <!DOCTYPE hibernate-mapping PUBLIC 
 3     "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
 4     "http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
 5     
 6 <hibernate-mapping package="day.one">
 7     <class name="Person" table="t_Person">
 8         <id name="id" column="id" unsaved-value="-1">
 9                 <!--主建生成策略  -->
10             <generator class="sequence">
11                  <param name="sequence">Person_id_seq</param>
12              </generator>
13         </id>
14      <!--当属性名与表的字段名相同时,可以确省column属性配置-->
15      <property name="name" column="name" />  
16      <property name="age" />
17      <property name="phone"/>
18     </class>
19 </hibernate-mapping>

    3、创建数据表与主键生成的Sequence
        上面Person这个JavaBean已经建立好了,并写好了相应的配置文件,现在就需要把对应的数据库表建立好,我使用的是Oracle数据库,建表和建立主键生成的Sequence的命令如下:
 1 --创建t_Person表
 2 create table t_Person
 3 (
 4  id number(7primary key,
 5  name varchar2(30not null,
 6  age number(4not null,
 7  phone varchar2(20)
 8 );
 9 
10 --创建t_Person的主键生成Sequence
11 create sequence person_id_seq;
12 select * from t_person;

        4、编写Hibernate配置文件(hibernate.cfg.xml)-->详细配置格式可查看Hibernate包下的src/org/hibernate/hibernate-configuration-3.0.dtd文件,里面有详细的描述。下边是Hibernate的配置文件的简单结构:
1 <hibernate-configuration>
2    <session-factory>
3      <property name="connection.driver_class">oracle.jdbc.driver.OracleDriver</property>
4            <!--详细配置信息可查看Hibernate包下的etc/hibernate.properties -->
5    </session-factory>
6 <hibernate-configuration>

        下面便是我配置的一个完整(相对完整,仅供参考)的Hibernate的配置文件,如下:
Hibernate配置文件

        4、创建配置(Configuration)
             Configuration cfg = new Configuration().configure();
     
         5、 构建会话工厂(SessionFactory)
             SessionFactory = cfg.buildSessionFactory();
     
          6、打开会话(session)
              Session session = sf.openSession();
      
          7、启动事务(Transaction)
              Transaction ts = session.beginTransaction();

          8、持久化操作(CUQD)
              session.save(*);  *表示一个对象
              session.update(*);
              session.delete(*);
              ..........................................
     
          9、提交事务(commit)
                  ts.commit();
      
          10、事务回滚(rollback)
                 ts.rollback();

四、第一个Hibernate程序
需要持久化的实体类

Hibernate映射文件

Hibernate配置文件

操作客户端

五、Hibernate映射过程分析
        如上就完成了对一个简单的对象的持久化操作,将构造好的Person对象通过Hibernate持久化操作到数据库。OK,下面我们来简单的分析下Hibernate做持久化的过程,他是怎么来完成映射的呢?
1 11:29:36,562  INFO HbmBinder:300 - Mapping class: day.one.Person -> t_Person
2 11:29:36,593 DEBUG HbmBinder:1270 - Mapped property: id -> id
3 11:29:36,609 DEBUG HbmBinder:1270 - Mapped property: name -> name
4 11:29:36,609 DEBUG HbmBinder:1270 - Mapped property: age -> age
5 11:29:36,609 DEBUG HbmBinder:1270 - Mapped property: phone -> phone

        上面这几行输入信息是在运行的时候输出的,详细可以查看控制台的输出信息。我们可以看出,Hibernate他在中间做了些什么工作呢?
        1、将day.one.Person这个类映射到了数据库的t_Person表,而这正是我们在映射文件中通过table属性来配置的,如下,将Person这个持久化类映射到数据库的t_Person表:
1 <class name="Person" table="t_Person">

        2、后面四行DEBUG输出的内容我想就不用多说了,也就是把映射文件里配置的property影射到数据库指定的字段:
1<!--id特殊,不使用property,hibernate为id的映射提供了专用的标记"id"-->
2<id name="id" column="id" unsaved-value="-1">
3
4<!--当属性名与表的字段名相同时,可以确省column属性配置-->
5<property name="name" column="name" />  
6<property name="age" />
7<property name="phone"/>
8
9

六、Hibernate持久化原理分析
        我们可以查看程序运行时输入的DEBUG信息,下面是截取的一部分:
1 selectselect id from t_Person where id =?
2 selectselect person_.id, person_.name as name0_, person_.age as age0_, person_.phone as phone0_ from t_Person person_ 
                where
 person_.id=?
3 Insert 0insert into t_Person (name, age, phone, id) values (?, ?, ?, ?)
4 Update 0update t_Person set name=?, age=?, phone=where id=?
5 Delete 0delete from t_Person where id=?
        当Hibernate成功完成映射后,表会为我们的持久化对象生成如上的相关SQL语句,这一步就是为后面做持久化操作做准备。我们在映射配置文件里指定的id生成策略为Sequence,这时Hibernate 就会先到数据库里去将指定的sequence查询出来赋值给持久化对象的id属性,然后在做其他的持久化操作。生成的查询语句如下:
 1 SQL:393 - 
 2     select
 3         Person_id_seq.nextval 
 4     from
 5         dual
 6 Hibernate: 
 7     select
 8         Person_id_seq.nextval 
 9     from
10         dual

        最终做持久化操作的时候便会条用上面所生成的SQL语句,Hibernate的底层在去通过JDBC完成数据库的相关操作。就本文前的实例程序,我们是将构造的Person对象持久化到数据库,那么他调用的SQL就如下:
 1SQL:393 - 
 2    insert 
 3    into
 4        t_Person
 5        (name, age, phone, id) 
 6    values
 7        (?, ?, ?, ?)
 8Hibernate: 
 9    insert 
10    into
11        t_Person
12        (name, age, phone, id) 
13    values
14        (?, ?, ?, ?)


        OK,下面我们在来做一个查询的操作,将刚刚通过session.save(person)到数据库的这条记录查询出来,看看DEBUG出的SQL语句是什么样的:

 1 SQL:393 - 
 2     select
 3         person0_.id as id0_0_,
 4         person0_.name as name0_0_,
 5         person0_.age as age0_0_,
 6         person0_.phone as phone0_0_ 
 7     from
 8         t_Person person0_ 
 9     where
10         person0_.id=?
11 Hibernate: 
12     select
13         person0_.id as id0_0_,
14         person0_.name as name0_0_,
15         person0_.age as age0_0_,
16         person0_.phone as phone0_0_ 
17     from
18         t_Person person0_ 
19     where
20         person0_.id=?

        现在就很容易看出了,无论是做什么样的持久化操作,他都是围绕着映射后所生成的SQL来处理的,其实这里的关键还是映射文件,映射文件配置好了,用Hibernate做持久化就不在是什么难事了。

本文就介绍于此,下一篇将介绍使用Hibernate来做一对一(OneToOne)的关系影射。

注:原创文章版权归作者,欢迎转载,未经作者同意必须注明文章出处或在页面适当位置给出原文连接,谢谢!
        文章出处:http://beniao.blogjava.net      作者:Beniao       时间:2008.08.14
        我的.NET技术博客:http://beniao.cnblogs.com