Hibernate View视图映射问题

今天遇到了一个 hibernate映射 视图的的问题 。 

这里简单的描述一下: 

使用hiberante 查询视图。 
在Hibernate中所有的实体都必须带有一个主键id 至少一个主键id。 
但是一个视图是不存在主键id的, 那么怎么办? 
很多人都是喜欢 随便找一个属性作为主键id, 这是不对的。 也是不正确的。 

我们来分析一下。 
如果使用hiberante的查询语句是 createSqlquery() 类型的语句,也就是sql语句,基本上是没有问题的. 但是你一但使用createQuery hql查询方式或者DetachedCriteria 查询方式就会出现问题. 
当你的视图数据有很多条重复的时候, hiberante会根据你设置的视图主键查询。 
例如: 
当你的视图主键是  名称这个属性(如下) 

序列号 名称    内容 
1     测试1   内容2 
2     测试1   内容3 

那么这个时候,hibernate会利用它强大的性能查询  找到第一条记录,然后 发现第一条记录的主键(测试1) 和第二天记录的主键(测试1)相同, hiberante过段的不进行查询了,直接把第第一条数据 拷贝到第二条数据。 

查询的结果是: 
名称     内容 
测试1   内容2 
测试1   内容2 

也许你会使用hibernate DetachedCriteria的detachedCriteria.setResultTransformer()这个方法设置去掉重复的, 但是这个时候, 已经得不到你想要的数据了。 

解决方案: 
我们在创建视图的时候 可以把视图的序列号 作为Id, 然后在视图实体里面 加上一个自增字段id 进行映射匹配!   
数据库视图 加上序号查询( oracle) 

select rownum as id from viewName 

完美解决 



一、思路

问题:

以往用Hibernate处理View时发生的困难主要集中在这两方面:

View是没有主键

由于View是由select语句生成,所以不存在任何主键。也就无法对其映射。

Hibernate必须对要操作的表进行po封装

Hibernate要求对数据库的操作都是通过pojo映射来实现的。也就是说无法对某个View直接读取而不对其进行映射。

解决:

要解决以上问题,我采取一下方案:

在建立View的时候生成类似主键的字段

通过其他手段建立类似主键的字段。

在映射po的时候手动把这个字段改为主键

将虚拟字段改为主键。根据虚拟的情况给以不同的自增长类型。

二、实现

建立视图

代码

  1. create or replace view v_user_perm   
  2. (id, userid, moduleid)   
  3. as   
  4. select rownum,a.id,c.id   
  5. from sysuser a,user_perm b,module c   
  6. where a.id=b.userid    
  7. and b.moduleid=c.id  

这个视图是从两个表中分别取出id,然后我将生成的视图的行号作为id处理。

建立xml

代码

  1. <?xml version="1.0" encoding="UTF-8"?>  
  2. <!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 2.0//EN" "http://hibernate.sourceforge.net/hibernate-mapping-2.0.dtd">  
  3. <hibernate-mapping>  
  4. <class name="com.fulong.journal.cad.po.perm.VUserPerm" table="V_USER_PERM" schema="JCAD">  
  5. <id name="id" column="ID" type="long">  
  6. <generator class="increment"/>  
  7. </id>  
  8. <property name="userid" column="USERID" type="string" length="32" not-null="true"/>  
  9. <property name="moduleid" column="MODULEID" type="string" length="32" not-null="true"/>  
  10. </class>  
  11. </hibernate-mapping>  

建立java

代码

  1.   
  2. import java.io.Serializable;   
  3. import org.apache.commons.lang.builder.ToStringBuilder;   
  4.   
  5. /** @author Hibernate CodeGenerator */  
  6. public class VUserPerm implements Serializable {   
  7.   
  8.     /** persistent field */  
  9.     private long id;   
  10.   
  11.     /** persistent field */  
  12.     private String userid;   
  13.   
  14.     /** persistent field */  
  15.     private String moduleid;   
  16.   
  17.     /** full constructor */  
  18.     public VUserPerm(String userid, String moduleid) {   
  19.         this.userid = userid;   
  20.         this.moduleid = moduleid;   
  21.     }   
  22.   
  23.     /** default constructor */  
  24.     public VUserPerm() {   
  25.     }   
  26.     public long getId() {   
  27.         return this.id;   
  28.     }   
  29.   
  30.     public void setId(long id) {   
  31.         this.id = id;   
  32.     }   
  33.   
  34.     public String getUserid() {   
  35.         return this.userid;   
  36.     }   
  37.   
  38.     public void setUserid(String userid) {   
  39.         this.userid = userid;   
  40.     }   
  41.   
  42.     public String getModuleid() {   
  43.         return this.moduleid;   
  44.     }   
  45.   
  46.     public void setModuleid(String moduleid) {   
  47.         this.moduleid = moduleid;   
  48.     }   
  49.   
  50.     public String toString() {   
  51.         return new ToStringBuilder(this)   
  52.             .toString();   
  53.     }   
  54.   
  55. }  
 注意:

view一般用在小涉及到多个表的组合查询时用到,用来提高效率!

view不能插入数据。


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值