使用Hibernate SQLquery实现动态表

开发 后端
本文向您介绍使用Hibernate SQLquery实现动态表的思路和实现方法,通过Hibernate SQLquery可以很好的解决问题,并且可以重新用于Hibernate hbm机制。

在实际的项目应用中,有时会设计出这样的一种数据表,每个时间段产生一个新表,例如是按年或月或日。相同类型的表中,所有的字段结构都是一样的。而 Hibernate 提供的类与表的映射,是只能映射到一个具体表的,在程序的运行过程中,很难去动态修改一个 hbm 对应的表名。我在网上也有看到一实现,但是很复杂,并且不符合我的要求。

因此我就想到直接用 jdbc 去操作数据库,这样的做法是绕过 Hibernate 了。方法是从 Hibernate 的 session 中,直接取得数据库 connection ,然后就直接 jdbc 了。

后来在升级了 proxool 到 9.0RC3 后,发现居然出现了数据库连接无法释放的问题。为了解决这个问题,我查阅了 Hibernate doc。我发现原来用Hibernate SQLQuery 可以更好的解决,并且可以重新用于 Hibernate hbm 机制。以下举例说明。

例如我有一个 pojo 是 ReadInfo,用来记录阅读信息的。由于数据量宠大,所以我的思路是按月划分,每个月一张表。所以只是表名不同,而字段是完全相同的。

ReadInfo.java 是这样的,其中 userId, year, month, day 是联合主键:

private Integer userId;

private Integer year;

private Integer month;

private Integer day;

private Integer point ;

那么相应的 ReadInfo.hbm.xml 的片段是

  1. < class name"ReadInfo" table"tblReadInfo " mutable = "false" >   
  2.         < composite-id>   
  3.             < key-property name"userId" column"userId" type"integer" / >   
  4.             < key-property name"year" column"year" type"integer" / >   
  5.             < key-property name"month" column"month" type"integer" / >   
  6.             < key-property name"day" column"day" type"integer" / >   
  7.         < / composite-id>   
  8.         < property name"point" column"point" type"integer" / >   
  9.     < / class>  

上面的xml,注意 2 个细节

1. pojo 所映射的 table tblReadInfo 实际上是不存在的。实际的表是 tblRead200710 之类的;

2. mutable 要设置为 false,即是说,关闭 Hibernate 对这个 pojo 的任何持久化操作,以避免 Hibernate 把数据写到 tblReadInfo 中(这个表是不存在的嘛)。因此,所有的持久化操作,都是需要自己通过Hibernate SQLQuery 来处理。

现在可以看一下 ado 中的操作了,先看一个 select 操作

  1. public ReadInfo selectReadInfo( Integer userId, Integer year,   
  2.             Integer month, Integer day) throws HibernateException  
  3.     {   
  4.          ReadInfo readInfo = null ;   
  5.  
  6.          Session session = getSession ( ) ;   
  7.         Transaction tx = session. beginTransaction( ) ;   
  8.  
  9.         try   
  10.         {   
  11.             String sql = "select * from tblRead"   
  12.                 + Misc. formatMoon( year, month)   
  13.                 + " where userId=? and day=?" ;   
  14.  
  15.              SQLQuery query = session. createSQLQuery( sql ) ;   
  16.             query . addEntity( ReadInfo. class ) ;   
  17.  
  18.             query . setLong ( 0, userId) ;   
  19.             query . setInteger( 1, day) ;   
  20.  
  21.              readInfo = ( ReadInfo) query . uniqueResult( ) ;   
  22.  
  23.              tx. commit ( ) ;   
  24.         }   
  25.         catch ( HibernateException e)   
  26.         {   
  27.             log . error ( "catch exception:" , e) ;   
  28.  
  29.             if ( tx ! = null )   
  30.             {   
  31.                  tx. rollback ( ) ;   
  32.             }   
  33.  
  34.             throw e;   
  35.         }   
  36.         return readInfo;   
  37.     }  

上面的代码,关键是以下几点:

1. 通过函数参数的 year, month 来确定要操作的表名,我自己写了一个 Misc.formatMoon(year, month) 来生成 "yyyyMM" 格式的字串;

2. 使用了 SQLQuery ,再通过 query.addEntity(ReadInfo.class); 建立与 ReadInfo 的映射关系;

3. query.setXxx() 与 PreparedStatement 的类似,不过索引是从 0 开始;

4. 其它的就跟一般的 Query 操作类似的了。

再看一个 insert 操作

  1. public void insertReadInfo( ReadInfo readInfo) throws HibernateException  
  2.     {   
  3.          Session session = getSession ( ) ;   
  4.         Transaction tx = session. beginTransaction( ) ;   
  5.  
  6.         try   
  7.         {   
  8.             String sql = "insert into tblRead"   
  9.                 + Misc. formatMoon( readInfo. getYear ( ) , readInfo. getMonth ( ) )   
  10.                 + " (userId, year, month, day, point) values (?, ?, ?, ?, ?)" ;   
  11.  
  12.              SQLQuery query = session. createSQLQuery( sql ) ;   
  13.  
  14.             query . setLong ( 0, readInfo. getUserId( ) ) ;   
  15.             query . setInteger( 1, readInfo. getYear ( ) ) ;   
  16.             query . setInteger( 2, readInfo. getMonth ( ) ) ;   
  17.             query . setInteger( 3, readInfo. getDay ( ) ) ;   
  18.             query . setInteger( 4, readInfo. getPoint ( ) ) ;   
  19.  
  20.             query . executeUpdate ( ) ;   
  21.  
  22.              tx. commit ( ) ;   
  23.         }   
  24.         catch ( HibernateException e)   
  25.         {   
  26.             log . error ( "catch exception:" , e) ;   
  27.  
  28.             if ( tx ! = null )   
  29.             {   
  30.                  tx. rollback ( ) ;   
  31.             }   
  32.  
  33.             throw e;   
  34.         }   
  35.     }  

同理,update, delete 等操作也是这样通过Hibernate SQLquery来实现的。

这种Hibernate SQLquery处理方式的麻烦的地方是需要手工写 SQL,因此要尽量写通用的标准SQL,不然在数据库兼容方面会有问题。当然,有时是会出现无法兼容的情况,那么可以考虑把 SQL写到配置文件中,根据不同的数据库,装载相应的配置文件。

 

【编辑推荐】

  1. Hibernate更新出错的解决
  2. 深入浅出Hibernate学习笔记 Criteria Query
  3. 正确理解Hibernate Inverse
  4. 对Hibernate缓存机制的分析
  5. Hibernate HQL查询解析
责任编辑:佚名 来源: CSDN
相关推荐

2011-04-07 14:04:28

SQL动态交叉表

2009-06-26 10:12:00

Hibernate自动

2009-09-25 10:38:42

Hibernate动态

2018-06-04 16:20:56

Linux动态路由Quagga

2010-11-12 13:27:13

动态sql

2009-09-14 19:20:22

LINQ TO SQL

2009-09-28 13:43:28

使用Hibernate

2009-09-23 10:14:10

Hibernate使用

2009-09-29 16:48:42

Hibernate J

2009-09-25 13:55:05

Hibernate使用

2009-06-30 16:52:30

DetchedCrit

2009-09-23 10:28:49

使用Hibernate

2009-09-21 17:23:49

Hibernate使用

2009-06-02 17:27:28

Hibernate框架ORM

2009-06-25 16:49:24

Hibernate

2009-09-25 11:14:16

Hibernate批量

2009-06-29 09:00:42

Hibernate的Q

2009-09-22 13:41:10

直接使用Hiberna

2011-03-24 11:37:41

Hibernate

2012-02-08 11:01:53

HibernateJava
点赞
收藏

51CTO技术栈公众号