框架的诞生必然是为了解决某一问题,要对Hibernate的身世有所了解,知道它的优点、作用,能够干什么,不能够干什么,这然才算是掌握了Hibernate.
典型的B/S三层架构,这个都不陌生:
为什么要把Dao单独作为一个层?这样设计肯定是有原因的.底层数据库的实现不同,Dao的实现也不尽相同,可能一个Dao接口下有很多个实现,比如MysqlDaoImpl或者OracleDaoImpl等,这样写很不方便,很麻烦,而且就算用dbUtils这样的框架简化开发,也有一些问题解决不了,比如要把一个域对象映射到一张表里面去,如果这个对象的属性名或者表的字段名发生了更改,就全部都要改,很不方便,而Hibernate就能解决这样的问题,它不用管底层数据库的实现是什么,只需要告诉它所说的方言并通过域对象生成SQL语句,通过配置文件映射到表,为业务逻辑层提供了面向对象的API,对数据访问进一步的封装.所以Hibernate的特点如下:
在实际应用中,对象的映射文件后缀名是.hbm.xml,Hibernate的主配置文件名为hibernate.cfg.xml.
映射文件描述对象与表之间的关系,映射文件并没有创建实质性的东西,只有当hibernate的Configuration类将配置文件读到流中,创建SessionFactory,SessionFactory才会检查表的结构,然后根据主配置文件里的hbm2ddl.auto的取值,作相应的反应.配置映射文件的时候,需要思考,需要什么信息,且只需要配置hibernate猜不到的信息,比如配置了对象的属性ID为主键,如果表中的主键列的列名为ID,则可以不用配置,不同的集合需要配置不同的属性:
Set
<set> 属性名、集合表、集合外键、存放元素的列
List
<list> 属性名、集合表、集合外键、存放元素的列、表示索引的列
Bag
<bag> 属性名、集合表、集合外键、存放元素的列
Map
<map> 属性名、集合表、集合外键、存放元素的列、表示key的列
数组
<array> 属性名、集合表、集合外键、存放元素的列、表示索引的列
集合有懒加载,是配置集合的元素中的lazy属性,可取的值有:true,false,extra.默认值为true,为懒加载,就是当你用的时候才加载.extra就是增强的"超级懒加载",此时大多数操作都不会初始化集合类,只有当程序第一次访问集合属性的iterator()方法时,才会导致集合类的初始化,或者是有时候只想看集合的长度或者是不是空的,不关心集合中存放着什么数据,如果这个时候把集合的全部数据都读取出来,就很没有必要,于是就有第二个作用:当程序第一次访问集合属性的 size(), contains() 和 isEmpty() 方法时, Hibernate 不会初始化集合类的实例, 仅通过特定的 select 语句查询必要的信息.需要注意的是只有集合的lazy属性才有extra的取值.普通的值映射如下:
<!-- package属性,表示当前配置中所写的类名如果没有包名,则默认是这个包中的。 --> |
<hibernate-mapping> |
<!--class元素表明哪个实体对应哪张表,一个class元素代表一个实体的映射 |
name:指定实体(类的全限定名) |
table:指定实体对应的表,可以不写,不写时代表表名和对象的简单名称一致 |
mutable:默认为true,表明该类的实例是可变的或者不可变的 |
dynamic-update:默认为 false,用于 UPDATE 的 SQL 将会在运行时动态生成,并且只更新那些改变过的字段 |
dynamic-insert:默认为 false,用于 INSERT 的 SQL 将会在运行时动态生成,并且只包含那些非空值字段 |
--> |
<class name="..domain.Person" table="person"> |
<!-- 主键映射 必须要有 --> |
<id name="id" type="int"> |
<!-- 主键的值生成器,有多种生成策略 --> |
<generator class="identity"></generator> |
</id> |
<!-- 值映射 |
name:代表实体的属性名 |
type:值类型,在数据库中一个列可以存放的属性,例:int, varchar, date) |
not-null:非空约束 |
--> |
<property name="name" type="string" length="12" not-null="true" /> |
<property name="birthday" type="date" /> |
<!-- 集合映射 |
bag map set list array对应不同的标签 |
--> |
<list name="interest" table="interestList"> |
<key column="interestID"></key> |
<!-- List需要顺序 --> |
<list-index column="index_"></list-index> |
<!-- 元素列 --> |
<element column="interest" type="string" length="22"></element> |
</list> |
</class> |
</hibernate-mapping> |
主键值的生成策略虽然有很多,但是只有当<generator>元素的class属性的值为assigned策略时才需要自己指定主键值,其他的都由Hibernate指定,自己不需要指定,指定了也没用还需要注意一点的就是increment,它是由Hibernate来维护的自增长,有并发问题,除了这种自增长,还有几种方式,native会根据数据库的能力选择 identity
、sequence
或者hilo
中的一个,如果是mysql就会选择identity,由数据库维护,不会有并发问题.如果生成策略是hilo(高/低算法),需要额外配置几个参数.主键的类型分为两种:自然主键和代理主键.自然主键是在数据库表中把具有业务逻辑含义的字段作为主键,比如有客户的姓名,把客户的姓名当作主键,就是自然主键;代理主键就是采用一个与当前表中逻辑信息无关的字段作为其主键,比如在一张表中插入一列与业务数据毫无关系的数据.
主配置文件可以包含三类信息的配置:数据库连接信息、其他配置和声明映射文件.例如:
<hibernate-configuration> |
<session-factory> |
<!-- 数据库连接信息 --> |
<property name="hibernate.connection.username">root</property> |
<property name="hibernate.connection.password">root</property> |
<property name="hibernate.connection.url">jdbc:mysql://localhost:3306/test</property> |
<property name="hibernate.connection.driver_class">com.mysql.jdbc.Driver</property> |
<property name="hibernate.dialect">org.hibernate.dialect.MySQLDialect</property> |
<!-- 其他配置 --> |
<property name="show_sql">true</property> |
<!-- 自动生成表结构,可选择的值有: |
create:每次启动时创建表结构,在建表前会先执行删除表的语句,以保证创建成功。 |
update:每次启动时检测一下,如果表结构没有,则创建,不一样,则更新。 |
create-drop: 每次启动时创建表结构,退出前删除表结构 |
validate: 每次启动时检测一下,如果表结构不一样,就报错 但是hbm2ddl.auto属性的配置一般在新增时有效,更新时无效.需要注意一下. |
--> |
<property name="hbm2ddl.auto">update</property> |
<!-- 声明映射文件 --> |
<mapping resource="../../domain/Person.hbm.xml" /> |
</session-factory> |
</hibernate-configuration> |
主配置文件相对简单,以后数据库的连接信息也不用放在主配置文件中.主配置文件里面配置了一个sessionFactory,Hibernate把一次数据库的访问当作是会话来看待,我们就需要在程序中获得会话,就要利用到sessionFactory,sessionFactory由主配置文件来描述,我们就先要读取调查主配置文件,然后由Hibernate来产生sessionFactory:
private static SessionFactory sessionFactory = new Configuration() |
.configure() |
.buildSessionFactory(); |
读取主配置文件的方法有多种,都是Configuration类的configure()方法重载的版本,最简单的就是将主配置文件命名为hibernate.cfg.xml,原因如下:
public Configuration configure() throws HibernateException { |
configure( "/hibernate.cfg.xml" ); |
return this; |
} |
这是源码.然后我们在主配置文件中利用<mapping>标签声明了映射文件,这个标签会找到指定的类,然后把相应的映射文件加载进来,我们还有另外一种方法加载映射文件:
private static SessionFactory sessionFactory = new Configuration() |
.configure() |
.addClass(Person.class) |
.buildSessionFactory(); |
只是比之前的代码多了一行,addClass()方法的返回值仍是Configuration对象,意味着如果你还有映射文件没有加载进来,仍可能继续接着写,那么它是如何找到映射文件的呢?其实很简单,我们一样可以写出来:
public Configuration addClass(Class persistentClass) throws MappingException { |
String mappingResourceName = persistentClass.getName().replace( '.', '/' ) + ".hbm.xml"; |
log.info( "Reading mappings from resource: " + mappingResourceName ); |
return addResource( mappingResourceName, persistentClass.getClassLoader() ); |
} |
这是它的源码,先得到类的全限定名,再把点替换把/,就得到了路径,再连接后缀,就行了,方法本身很简单,我们也要掌握它的思想,也就是方法调用链.