当咱们用HQL进行子查询的时候,如select * from Tree where pid in (select id from Tree,此时HIBERANTE就会报错,说什么*号错误之类的。但若是将*改成Tree类里的全部子段时就不会有问题了。就会像平时同样第一行数据返回一个Object[],而后你再根据Tree类里字段对Object[]数组里的值进行转换。这样一来比较麻烦。今天发现若是我SQL来查有一个方法能够返回一个对象的。
Configuration config = new Configuration().configure();
SessionFactory sf = config.buildSessionFactory();
Session session = sf.openSession();
Transaction ts = session.beginTransaction();
Query query = session.createSQLQuery("select * from Tree t where pid in (select id from Tree) ").addEntity(Tree.class); //返回对象
List list = query.list();
此时在遍历list时就能够(Tree)list.get[i];将每一行的内容变换为一个对象了。
另还能够返回一个Map对象,也就是说在在list里包含多个Map,代码以下
Query query = session.createSQLQuery("select id,name from Tree t where pid in (select id from Tree) ").setResultTransformer(Transformers.ALIAS_TO_ENTITY_MAP); //返回一个map,KEY:为DB中名称一致(大小写一致)遍历list时就能够
Map map = (Map)list.get[i];
map.get("id");map.get("name");来取值。按你的SQL语句select后的字段名来做为map的Key,但这个key必须与数据库中的字段名如出一辙。
还能够用做函数方面的。如
Query query = session.createSQLQuery("select sum(id) SUMID from Tree t where pid in (select id from Tree)
.addScalar("SUMID",Hibernate.INTEGER) //转换类型,按DB中的type转
.setResultTransformer(Transformers.ALIAS_TO_ENTITY_MAP); //返回一个map,KEY:为DB中名称一致(大小写一致)
直接就map.get("SUMID")能够取值了
还有一点就是这个方法在Hibernate3.2版本上才能正常运行。
=================================
问题:
仔细查看,发现问题在于数据类型.到网上查,发现hibernate在执行List result = session.createSQLQuery(sql).list()的时候,当SQL语句中遇到的decimal,long等类型的字段时,就出现上面的错误.并且从错误信息中能够发现:出错的是Dialect.
解决办法:
错误知道之后,我就到网上找解决办法.看来遇到这类问题的人太多了,网上处处都有人贴这个问题.我看了几篇,发现有个解决办法,就是自定义Hibernate Dialect.虽然所用数据库不一样(我用的数据库是DB2),我以为大同小异,就照着作了:
首先建一个类,继承org.hibernate.dialect.DB2Dialect,该类的内容以下:
import java.sql.Types;
import org.hibernate.Hibernate;
import org.hibernate.dialect.DB2Dialect;
public class PmDb2Dialect extends DB2Dialect
{
public PmDb2Dialect()
{
super();
registerHibernateType(Types.DECIMAL, Hibernate.BIG_DECIMAL.getName());
}
}
第二步,就是修改hibernate的配置文件hibernate.cfg.xml:
将:
<property name="hibernate.dialect">
org.hibernate.dialect.DB2Dialect
</property>
改成:
<property name="hibernate.dialect">
com.yonder.pm.common.PmDb2Dialect
</property>
===============================
昨天遇到问题,以下:
ORACLE数据库中,字段类型CHAR(8),值12345678
hibernate中用createSQLQuery方法查询,返回的list用object[]接收,遍历取值发现object[0]输出值是1,只有一位,其余的没了。其余字段正确。
---------------------------------------------
查看数据库,发现其余字段包括VARCHAR,DATE等类型均无问题,只有char类型的出问题。
char类型是定义长度的,8表明8个字节,节省空间而且效率要高,缺点是不灵活,长度是定死的,这里用来定义站号,固定8位长度。因此,该数据库这个字段类型能解决问题,但不是最好的办法,也没找到真正缘由。
-----------------------------------------------
查到如今,有了一些眉目,小结以下:
1,oracle的char字段在hibernate里映射为character类型,是varchar的子集。
2,复杂SQL用createSQLQuery方法查询没问题,若是查询多个字段,遍历用object[]造型,下标从0开始输出值,不须要映射文件;若是愿意能够写一个映射bean,方便取用。
3,若是查询SQL中是只有一个字段,那就不能用object[]数组接收,只能用object类接收,直接输出object.toString(),便是这个字段的值。
4,能够用addScalar(String arg,Type type)方法定义要返回的字段类型,如
s.createSQLQuery(shuiQingHQL).addScalar("STCD",Hibernate.STRING).addScalar("STNM");
这样就解决了CHAR字段类型只出一位字符的问题。
可是须要把其余字段也addScalar()进来!
java