Hibernate对原生sql处理及结果集报错:Expected type: java.lang.Integer, actual value: java.math.BigInteger

基于Hibernate执行sql查询方法,映射实体对象报错Expected type: java.lang.Integer, actual value: java.math.BigInteger

实体类ChannelTree

package com.han.cms.dto;

import java.math.BigInteger;

/**
 * 系统栏目树对象
 * @author Administrator
 *
 */
public class ChannelTree {
	private Integer id;
	private String name;
	private Integer pid;//映射报错,将Integer类型改成BigInteger类型即可

	public ChannelTree() {

	}

	public ChannelTree(Integer id, String name, Integer pid) {
		this.id = id;
		this.name = name;
		this.pid = pid;
	}

	public Integer getId() {
		return id;
	}

	public void setId(Integer id) {
		this.id = id;
	}

	public String getName() {
		return name;
	}

	public void setName(String name) {
		this.name = name;
	}

	public Integer getPid() {
		return pid;
	}

	public void setPid(Integer pid) {
		this.pid = pid;
	}

}

service查询方法

Override
	public List getChannelTreeDatas() throws Exception{
		String sql="select c.id,c.name,IFNULL(c.pid,0) as pid from t_channel c";
		List channelTrees = this.channelDao.findListBySql(sql, ChannelTree.class, false);
		channelTrees.add(0,new ChannelTree(0,"系统栏目管理",new BigInteger("-1")));
		return channelTrees;
	}


test测试类方法

@Test
	public void getChannelTrees() throws Exception{
		System.out.println(this.channelService.getChannelTreeDatas().size());
	}

运行结果报错:

Hibernate: select c.id,c.name,IFNULL(c.pid,0) as pid from t_channel c
15:39:55,758 ERROR BasicPropertyAccessor:117 - HHH000123: IllegalArgumentException in class: com.han.cms.dto.ChannelTree, setter method of property: pid
15:39:55,760 ERROR BasicPropertyAccessor:118 - HHH000091: Expected type: java.lang.Integer, actual value: java.math.BigInteger

修改之后ChannelTree

package com.han.cms.dto;

import java.math.BigInteger;

/**
 * 系统栏目树对象
 * @author Administrator
 *
 */
public class ChannelTree {
	private Integer id;
	private String name;
	private BigInteger pid;

	public ChannelTree() {

	}
    
	public ChannelTree(Integer id, String name, BigInteger pid) {
		this.id = id;
		this.name = name;
		this.pid = pid;
	}

	public Integer getId() {
		return id;
	}

	public void setId(Integer id) {
		this.id = id;
	}

	public String getName() {
		return name;
	}

	public void setName(String name) {
		this.name = name;
	}

	public BigInteger getPid() {
		return pid;
	}

	public void setPid(BigInteger pid) {
		this.pid = pid;
	}
    
}
service查询方法

	@Override
	public List getChannelTreeDatas() throws Exception{
		String sql="select c.id,c.name,IFNULL(c.pid,0) as pid from t_channel c";
		List channelTrees = this.channelDao.findListBySql(sql, ChannelTree.class, false);
		channelTrees.add(0,new ChannelTree(0,"系统栏目管理",new BigInteger("-1")));
		return channelTrees;
	}


测试类

@Test
	public void getChannelTrees() throws Exception{
		System.out.println(this.channelService.getChannelTreeDatas().size());
	}

运行结果:

Hibernate: select c.id,c.name,IFNULL(c.pid,0) as pid from t_channel c
4

具体的解释可以参考以下的

我有一个类似下面的table(info_table),是收集产品使用的机型信息:

id

type

model

1

nokia

xxx1

2

nokia

xxx2

3

Motorola

yyyy1

4

Motorola

yyyy2

5

Motorola

yyyy3

要实现一个前台展示的页面:

type

countType

nokia

2

Motorola

3

就这么简单的功能。相信稍微熟悉sql的人,马上就可以写出此实现。很惭愧的说,我的心里感觉很简单,由于很久没怎么深入接触sql了,也费了一番周折...

selecttype,count(*) as countType from info_table group by Type order bycountType desc

实现上面的功能。

如何放到功能(struts2.3.x,spring3.1.x,hibernate4.x)集成的环境中处理?一步一步看:

首先从dao层:

[java]  view plain  copy
  1. @Override  
  2. publicList getAllHandleCount(){  
  3. Sessionsession =sessionFactory.getCurrentSession();  
  4. Stringhql ="selecttype,count(*) as countType from info_table group by Type order bycountType desc";  
  5. Queryquery = session.createQuery(hql);  
  6. @SuppressWarnings("unchecked")  
  7. Listlist = query.list();  
  8. returnlist;  
  9. }  

再次server...

最后action层;

[java]  view plain  copy
  1. @Autowired  
  2. privateHandleCountServicehandleCountService;  
  3. privateIntegercurrentPage= 1;  
  4. privateIntegerpageSize= 10;  
  5. privateIntegertotalCount;  
  6. privateIntegertotalPage;  
  7.   
  8. privateListhandleCountVO;  
  9. /* 
  10. * handle count at 2014-12-04 
  11. * 
  12. */  
  13. publicString HandleCountQueryAll() throwsIOException{  
  14. handleCountVO=handleCountService.getAllPage(currentPage,pageSize);  
  15. totalCount=handleCountService.getAllHandleCount().size();  
  16. totalPage= (totalCount+pageSize- 1) /pageSize;  
  17. returnSUCCESS;  
  18. }  
  19. ……  
  20. get/set方法  

由于返回的结果不是Hibernate管理的bean,所以理所当然的想到写个VO去接纳返回结果集。

CountVO.java

[java]  view plain  copy
  1. publicclassCountVO {  
  2. privateString type;  
  3. privateIntegercountUser;  
  4. publicString getType(){  
  5. returntype;  
  6. }  
  7. publicvoidsetType(Stringtype){  
  8. this.type=type;  
  9. }  
  10. publicInteger getCountUser() {  
  11. returncountUser;  
  12. }  
  13. publicvoidsetCountUser(Integer countUser) {  
  14. this.countUser= countUser;  
  15. }  
  16. }  

前台直接通过传递ListBean通过struts标签来实现:

xxxx.jsp

[html]  view plain  copy
  1. …  
  2. <s:iteratorvalues:iteratorvalue="handleCountVO">  
  3. <tr>  
  4. <td><s:propertyvalues:propertyvalue="type"/>td>  
  5. <td><s:propertyvalues:propertyvalue="countUser"/>td>  
  6. tr>  
  7. s:iterator>  
  8. …  

整个流程配置完成。对我来说看似没问题,但是实际不是这样的...…


问题1

action层明明看到有list值,传到jsp层就是不显示,后debug跟到jsp,发现也是可以循环的,可“奇怪”的就是不显示。

经过一番折腾,才发现经过sql获得的List不是“理所当然“的List而是List,其里面的值不是我想的CountVO中的typecountUser,而是[0],[1]


需要找方法解决这个问题,很简单的一个解决方法-----转换一下就可以了,我不想这样做,想弄清楚这个问题。


由于没开,暂时baidu查查。

发现:

1)有个createSQLQuery和一直用的createQuery不同,因为CountVO不属于Hibernate管理的bean,不对应实体表,这个语句可能使用,最主要的还在后面;

createQuery用的hql语句进行查询,createSQLQuerysql语句查询;
前者以hibernate生成的Bean为对象装入list返回;
后者则是以对象数组进行存储;

所以使用createSQLQueryhibernate生成的Bean为对象装入list返回,可以直接转换对象 

Query query = session.createSQLQuery(sql).addEntity(XXXXXXX.class); 
XXXXXXX 
代表以hibernate生成的Bean的对象,也就是数据表映射出的Bean

(以上方法,只是介绍使用原生sql查询结果映射为hibernatebean方法,不能解决此问题)

2createSQLQuery后可以有这样的方法:setResultTransformer(...)


 Query

setResultTransformer(ResultTransformer transformer) 
          Seta strategy for handling the query results.

(只能google一下了)

Query setResultTransformer(ResultTransformer transformer)
Seta strategy for handling the query results. This can be used tochange "shape" of the query result.
Parameters:
transformer -The transformer to apply
Returns:
this(for method chaining)

这个函数功能:设置处理查询结果集的策略。

参数ResultTransformer定义:

public interface ResultTransformerextends Serializable

Implementorsdefine a strategy for transforming query results into the actualapplication-visible query result list.

setResultTransformer的执行者,转换查询结果到实际应用的结果列表。
ResultTransformer是个接口,通过hibernate3.3.xdoc文档可以看到其有多种实现类。

在一些博客中看到最多的是:Transformers.aliasToBean()

static ResultTransformer

aliasToBean(Class target) 
          Createsa result transformer that will inject aliased values intoinstances of Class via property methods or fields.

hibernate3.3.xdoc文档也看到:

org.hibernate.transform 
ClassAliasToBeanResultTransformer

java.lang.Object
  org.hibernate.transform.AliasToBeanResultTransformer


中构造函数:
AliasToBeanResultTransformer(Class resultClass) 

也可以满足需求。

那就开始做吧-------可以完美解决

问题2

sql转换的一个小问题,也列到这吧,不让问题1孤单的存在着:

前面"selecttype,count(*) as countType from info_table group by Type order bycountType desc";

countcountVOcountType类型问题

IllegalArgumentExceptionin class: com.xxx.bean.CountVO, setter method of property: countUser

2014-12-0511:39:51,708 ERROR HHH000091: Expected type:java.lang.Integer,actual value: java.math.BigInteger

2014-12-0511:39:52,337 ERROR Exception occurred during processing request:IllegalArgumentExceptionoccurred while calling setter of com.xxx.bean.CountVO.countUser

org.hibernate.PropertyAccessException:IllegalArgumentExceptionoccurred while calling setter of com.xxx.bean.CountVO.countUser

类型不匹配,sqlcount(*)需要对应类型为java.math.BigInteger,那就把CountVO.javacountUser转为其类型。

解决此问题。


参考:

https://docs.jboss.org/hibernate/core/3.3/api/org/hibernate/transform/ResultTransformer.html

https://docs.jboss.org/hibernate/core/3.3/api/org/hibernate/transform/AliasToBeanResultTransformer.html

http://blog.163.com/charm_888/blog/static/608350020107254126654/

http://langgufu.iteye.com/blog/1565397




你可能感兴趣的:(Hibernate)