<?xml version='1.0' encoding='GBK'?> <!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.url"> jdbc:mysql://localhost:3306/study </property> <property name="dialect"> org.hibernate.dialect.MySQLDialect<!-- 这里是MySQL的Dialect --> </property> <property name="myeclipse.connection.profile">MySQL5.0</property> <property name="connection.password">root</property> <property name="connection.driver_class"> org.gjt.mm.mysql.Driver<!-- 这里是MySQL的JDBC driver class名 --> </property> <property name="show_sql">true</property><!-- 向控制台输出SQL语句 --> <mapping resource="hibernate/PO/TRegister.hbm.xml" /><!-- 指定要用到的映射文件 --> </session-factory> </hibernate-configuration>
<?xml version="1.0" encoding="GBK"?> <!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN" "http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd"> <!-- 映射文件 --> <hibernate-mapping> <class name="hibernate.PO.TRegister" table="t_register"> <id name="id" type="java.lang.Integer"> <column name="id" /> <!-- 我在MYSQL中并没有设置该字段递增,但在Hibernate中依然可以设置为递增 --> <generator class="increment" /> </id> <property name="userName" type="java.lang.String"> <column name="userName" length="30" /> </property> <property name="userPwd" type="java.lang.String"> <column name="userPwd" length="30" /> </property> <property name="sex" type="java.lang.String"> <column name="sex" length="10" /> </property> <property name="age" type="java.lang.Integer"> <column name="age" /> </property> </class> </hibernate-mapping>
Session接口:
Session是应用程序与数据库之间的一个会话,是Hibernate运作的中心,持久层操作的基础,相当于JDBC中的Connection。Session对象是通过SessionFactory创建的:
Session session = SessionFactory.openSession();
一个持久化类与普通的JavaBean没有任何区别,但是它与Session关联后,就具有了持久化能力。当然,这种持久化操作是受Session控制的,即通过Session对象的装载,保存,创建或查询持久化对象。Session类的save(),delete()和load()等方法,来分别完成对持久化对象的保存,删除,修改加载等操作!Session类方法的用途可以分以下五类:
1:取得持久化对象:get()和load()等方法。
2:持久化对象的保存,更新和删除:save(),update()saveOrUpdate()和delete()等方法。
3:createQuery()方法:用来从Session生成的Query对象。
4:beginTransaction()方法:从Session对象生成一个Transaction对象。
5:管理Session的方法:isOpen(),flush(),clear(),evict()和close()等方法,其中isOpen()方法用来检查Session是否仍然打开;flush()用来清理Session缓存,并把缓存中的SQL语句发送出去,clear()用来清除Session中的所有缓存对象evict()方法来清楚Session缓存中的某个对象;close()关闭Session。
取得持久化对象的方法:
取得持久化对象的方法主要有get()和load(),它们通过主键id来取得PO。
get()方法示例
持久化对象的保存,更新和删除方法
save()方法:
session的save()方法将 一个PO的属性取出放入PreparedStatement语句中,然后向数据库中插入一条记录(或多条记录,如果有级联)。
session保存一个对象时,按如下步骤进行:
1:根本配置文件为主键id设置的生成算法 ,为PO指定 一个ID。
2:将 PO对象纳入session内部缓存(一个Map)内。
3:事务提交时,清理缓存,将 新对象通过insert语句持久化到数据库中。
如果要为新的PO强制指定一个ID,可以调用Session的重载方法save(Object obj,Serializable id)
例:
session.save(tRegister, new Integer(123));
在调用save()方法时,并不立即执行SQL语句,而是等到清理完毕缓存时才执行。如果在调用save()方法后又修改了PO的属性,则Hibernate将 会发送一条insert语句和一条update语句来完成持久化操作,如下:
监视上述程序运行会产生二条SQL:
Hibernate: select max(id) from t_register
Hibernate: insert into t_register (userName, userPwd, sex, age, id) values (?, ?, ?, ?, ?)
Hibernate: update t_register set userName=?, userPwd=?, sex=?, age=? where id=?
因此,最好是在对象状态稳定时再调用 save()方法,可以少执行一条update语句。
调用save()方法将临时对象保存到数据库中,对象的临时状态将 变为持久化状态。当对象在持久化状态时,它一直位于Session缓存中,对它的任何操作在事物提交时都将同步保存到数据库中,因此,对一个已经持久化的对象调用save()或update()方法是没有意义的,如下:
程序运行效果还是和上面的一样!
update()方法:
Session的update()方法是用来更新脱管对象的。它的用法如下:
监视运行:
Hibernate: select tregister0_.id as id0_0_, tregister0_.userName as userName0_0_, tregister0_.userPwd as userPwd0_0_, tregister0_.sex as sex0_0_, tregister0_.age as age0_0_ from t_register tregister0_ where tregister0_.id=?
Hibernate: delete from t_register where id=?
在上述代码中session2先把tr对象进行关联,纳入Session缓存中,然后删除。需要注意的是,在调用delete()方法时并不是发送SQL语句,而是在提交事务时,清理了缓存才发送SQL。
使用delete()删除对象时,会有一些性能上的问题,例如从上面的监视中可以看到,当删除一个对象时,会先调用get()加载这个对象,然后调用delete()方法删除对象,所以发送了一个多余的selete SQL,所以当删除大量数据时对性能影响就比较大了。为了解决批量删除的性能问题,常用的办法是使用批量删除操作,如下:
监视运行:
Hibernate: delete from t_register
只用了一条语句就完成了比量删除的操作。但它也有问题,批量删除后的数据还会存在缓存中,因此程序查询时可能得到脏数据(得到数据库中已不存在的数据),因此在使用批量删除时,也要经窒处世数据一致的问题。
Query接口:
在Hibernate3.x中不再使用2.x中的find方法,而是引入了Query接口,用来执行HQL。其实在上面的例子中已经看出,Query接口可以从session对象生成;
Query接口主要方法有三个
setXXX()方法:用于设置HQL中问题或变量的值。
list()方法:返回查询结果,并把结果转换成List对象。
executeUpdate()方法:执行更新或删除名。
setXXX()方法都有二种重载方法:
1:setString(int position,String value):用于设置HQL中“?”的值:其中position表示?位置,而value自然就是值。如下:
2:setString(String paraName,String value);用于设置HQL中“:”后跟变量的值;其中paraName代表HQL中“:”后跟变量,value为该变量设置的值。如下:
在HQL中使用变量代替问号,然后在setXXX()方法中为该变量设值。在HQL中使用变量相比问号有以下好处:
1:变量不依赖于它们在查询字符串中出现 的顺序。
2:在同一个查询中可以多次使用。
3:可读性好。
list()方法:
Query的list()方法用于取得查询结果,并将结果转变成一个List接口的实例。如下:
Trancation接口
该接口允许应用等量齐观定义工作单元,同时又可调用JTA或JDBC执行事物管理。它的运行与Session接口相关,可调用Session的beginTransaction()方法生成一个Transaction实例。
一个Session实例可以与多个Transaction实例相关联,但是一个特定的Session实例在任何时候必须与至少一个未提交的Transaction实例相关联。
Transaction接口常用如下方法:
1:commit();提交相关联的Session实例。
2:rollback();撤销事物操作。
3:wasCommitted();事物是否提交。
在本篇文章最后再介绍一下我在Hibernate框架简述中的HibernateUtil类
HibernateUtil类是Hibernate的Session管理的一个应用广泛的方案。它是使用ThreadLocal类建立的一个Session管理的辅助类。使用ThreadLocal可以有效隔离执行所使用的数据,所以避开了Session的多线程之间的数据共享问题。
本文借鉴:http://www.cnblogs.com/eflylab/archive/2007/01/09/615362.html