框架的诞生必然是为了解决某一问题,要对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() ); |
} |
这是它的源码,先得到类的全限定名,再把点替换把/,就得到了路径,再连接后缀,就行了,方法本身很简单,我们也要掌握它的思想,也就是方法调用链.
分享到:
相关推荐
Hibernate是一个开放源代码的对象关系映射框架(ORM),它对JDBC进行了非常轻量级的对象封装。
Hibernate 3.6 Final所有的jar包,以及Hibernate Tools 中的hibernate-tools.jar 和 freemarker.jar 。 本jar包用于使用hibernate-tools生成POJO所需要的依赖库
添加Hibernate3.6的核心类库,包含:hibernate3.jar、antlr-2.7.6.jar、cglib-nodep-2.1_3.jar、commons-collections-3.1.jar、dom4j-1.6.1.jar、ehcache-1.5.0.jar、hibernate-jpa-2.0-api-1.0.0.Final.jar、...
hibernate3.6.4.final.CHM hibernate的使用帮助文档
我见过的最好的最详细的hibernate3.6.X学习资料(汤阳光)
详细讲述了Hibernate3.6,里面大量的例子代码、图示和标注,对于Hibernate初学者是很好的学习资料。
这是开发hibernate框架的framework官方中文文档,可以参照此文档写hibernate的helloworld程序
hibernate3.6相关jar包,hibernate3.6相关jar包,hibernate3.6相关jar包
hibernate开发所需要的几个核心jar包,hibernate常用jar包,hibernate3.jar
hibernate3.6基本jar包
Hibernate3.6中文文档
hibernate3.6参考文档中英文
hibernate-distribution-3.6.10.Final
hibernate3.6 所有jar包
Hibernate3.6 api_en_US,Hibernate3.6最新英文版 chm格式api帮助文档
Hibernate 3.6 src 源代码 hibernate-core-3.6.0.Final.src.tar
hibernate3.6中文帮助文档
hibernate3.6 对应的 hibernate-validator-4.1.0
hibernate3.6完整的中文开发文档