今天遇到了一个 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
完美解决
这里简单的描述一下:
使用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的时候手动把这个字段改为主键
将虚拟字段改为主键。根据虚拟的情况给以不同的自增长类型。
二、实现
建立视图
代码
- create or replace view v_user_perm
- (id, userid, moduleid)
- as
- select rownum,a.id,c.id
- from sysuser a,user_perm b,module c
- where a.id=b.userid
- and b.moduleid=c.id
这个视图是从两个表中分别取出id,然后我将生成的视图的行号作为id处理。
建立xml
代码
- <?xml version="1.0" encoding="UTF-8"?>
- <!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 2.0//EN" "http://hibernate.sourceforge.net/hibernate-mapping-2.0.dtd">
- <hibernate-mapping>
- <class name="com.fulong.journal.cad.po.perm.VUserPerm" table="V_USER_PERM" schema="JCAD">
- <id name="id" column="ID" type="long">
- <generator class="increment"/>
- </id>
- <property name="userid" column="USERID" type="string" length="32" not-null="true"/>
- <property name="moduleid" column="MODULEID" type="string" length="32" not-null="true"/>
- </class>
- </hibernate-mapping>
建立java
代码
- import java.io.Serializable;
- import org.apache.commons.lang.builder.ToStringBuilder;
- /** @author Hibernate CodeGenerator */
- public class VUserPerm implements Serializable {
- /** persistent field */
- private long id;
- /** persistent field */
- private String userid;
- /** persistent field */
- private String moduleid;
- /** full constructor */
- public VUserPerm(String userid, String moduleid) {
- this.userid = userid;
- this.moduleid = moduleid;
- }
- /** default constructor */
- public VUserPerm() {
- }
- public long getId() {
- return this.id;
- }
- public void setId(long id) {
- this.id = id;
- }
- public String getUserid() {
- return this.userid;
- }
- public void setUserid(String userid) {
- this.userid = userid;
- }
- public String getModuleid() {
- return this.moduleid;
- }
- public void setModuleid(String moduleid) {
- this.moduleid = moduleid;
- }
- public String toString() {
- return new ToStringBuilder(this)
- .toString();
- }
- }
view一般用在小涉及到多个表的组合查询时用到,用来提高效率!
view不能插入数据。