在平时开发中Hibernate提供的hql基本能够满足我们的日常需求。但是在有些特殊的情况下,还是需要使用原生的sql,并且希望sql查询出来的结果能够绑定到pojo上。hibernate API中的createSQLQuery  和createQuery接口。
就像在这次的项目中,因为表结构要变化,有个新的需求:
要从一个表中查询极个别的字段并且还有几个是求多条数据的和的,之前一直在使用hibernate的API,使用的都是映射过得对象。
原对象及表结构是这样的:
@Entity
@Table(name="T_BILL_ACCT_ITEM")
@NamedQuery(name="TBillAcctItem.findAll", query="SELECT t FROM TBillAcctItem t")
public class TBillAcctItem implements Serializable {
private static final long serialVersionUID = 1L;
@Id
@Column(name="ACCT_ITEM_ID")
private long acctItemId; @Column(name="ACCT_ID")
private long acctId; @Column(name="BILLING_CYCLE_ID")
private String billingCycleId; @Column(name="PRODUCT_ID")
private String productId; @Column(name="SERVICE_ID")
private String serviceId; @Column(name="ACCT_ITEM_CODE")
private String acctItemCode; @Column(name="ORIGINAL_AMOUNT")
private int originalAmount; @Column(name="CDR_DISCOUNT")
private int cdrDiscount; @Column(name="ACCT_DISCOUNT")
private int acctDiscount; @Column(name="RECE_AMOUNT")
private int receAmount; @Column(name="REAL_AMOUNT")
private int realAmount; @Column(name="CHARGE_OFF_SOURCE")
private int chargeOffSource; @Column(name="STATE")
private int state;

但是我现在需要查询出来的信息是这样的一个pojo类:(因为int类型的值不能为空,所以我需要将字段定义为 java.long.Integer类型)

public class ProductBillInfo implements Serializable{

    private static final long serialVersionUID = 1L;
private String productId;//产品标识
private String serverId;//服务标示
private Integer realAmount;//实收金额(sum求和)
private Integer receAmount;//应收金额(sum求和)
private int state;//账目状态
private String billingCycle;//账期(月)

因为hibernate要查询的对象是要和表结构一一映射的,现在求和字段这些个是映射不了的, 之前的方法就有问题了。 找了很多看了hibernate的API 找到这个东西。  如下:

大概的使用过程:
        StringBuffer b = new StringBuffer();
//... 省略SQL拼接代码
String sql = b.toString();
SQLQuery query = session.createSQLQuery(sql);
query.addScalar("productId", StandardBasicTypes.STRING);//要查询出来的字段、类型
List<ProductBillInfo> lnfo =query.list();

下面是我用到的查询pojo的方法中的实现:

@SuppressWarnings("unchecked")
public List<ProductBillInfo> findByAcctIdAndCycle1(long acctId,
String cycleBegin, String cycleEnd, int state){ List<String> cycles = DateUtil.getAllCycle(cycleBegin, cycleEnd);
StringBuffer buf = new StringBuffer();
buf.append("select t.PRODUCT_ID as productId , t.SERVICE_ID as serviceId ," +
"sum(t.REAL_AMOUNT) as realAmount ,sum(t.RECE_AMOUNT) as receAmount ," +
"t.STATE as state ,t.BILLING_CYCLE_ID as billingCycleId " +
" from T_BILL_ACCT_ITEM t where t.STATE= '" + state + "' AND t.ACCT_ID='" + acctId + "' AND t.BILLING_CYCLE_ID in (");
StringBuffer bf = new StringBuffer();
for(String id : cycles){
if(bf.toString().equals("")){
bf.append("'"+id+"'");
} else{
bf.append(","+"'"+id+"'");
}
}
buf.append(bf.toString());
buf.append(") GROUP BY t.PRODUCT_ID,t.BILLING_CYCLE_ID,t.SERVICE_ID"); SQLQuery query = getSession().createSQLQuery(buf.toString());
query.addScalar("productId", StandardBasicTypes.STRING);
query.addScalar("serviceId", StandardBasicTypes.STRING);
query.addScalar("realAmount", StandardBasicTypes.INTEGER);
query.addScalar("receAmount", StandardBasicTypes.INTEGER);
query.addScalar("state", StandardBasicTypes.INTEGER);
query.addScalar("billingCycleId", StandardBasicTypes.STRING); query.setResultTransformer(Transformers.aliasToBean(ProductBillInfo.class));
List<ProductBillInfo> lnfo =query.list();
return lnfo;
}

注:示例中addScalar及setResultTransformer很关键,我在使用的时候还报了一个错(原因是在实体中需要有一个无参构造函数),否则错误如下:

org.hibernate.HibernateException: Could not instantiate resultclass:com.eastelsoft.framework.formbean.interfaceBean.ProductBillInfo
还需要说明的一点是
query.addScalar("deviceId", Hibernate.STRING);  ,老版本的使用的数据的类型都是 org.hibernate.type包下面的,
新版本的是在:org.hibernate.type.StandardBasicTypes包下面的
并且在语法中要使用到in(‘’、‘’、‘’)的语法,于是写了下面的一个拼接小方法:
public static void main(String[] args) {
List<String> c = new ArrayList<String>();
c.add("1");c.add("2");c.add("3");
StringBuffer bf = new StringBuffer();
for(String id : c){
if(bf.toString().equals("")){
bf.append("'"+id+"'");
} else{
bf.append(","+"'"+id+"'");
}
}
System.out.println(bf.toString());
}

打印结果是:     '1','2','3'   

使用StringBuffer拼接到原生的sql语句中

hibernate使用setResultTransformer()将SQL查询结果放入集合中的更多相关文章

  1. 在查询时将查询条件放入Session中,导出时直接根据qpniRGaFiler取查询条件即可

    在查询时将查询条件放入Session中,导出时直接根据qpniRGaFiler取查询条件即可

  2. java集合-把商品放入集合中调用(新手)

    //创建的一个包名. package qige; //导入的一个包.import java.util.*; //定义一个类.public class Ipcs { //公共静态的主方法. public ...

  3. java通过文件路径读取该路径下的所有文件并将其放入list中

    java通过文件路径读取该路径下的所有文件并将其放入list中   java中可以通过递归的方式获取指定路径下的所有文件并将其放入List集合中.假设指定路径为path,目标集合为fileList,遍 ...

  4. 用SQL查询方式显示GROUP BY中的TOP解决方法[转]

    用SQL查询方式显示GROUP BY中的TOP怎样用一个SQL语句来显示 分组后每个组的前几位 比如把一个学校所有学生的成绩按班级分组,再显示每个班级前五名的信息. 班级     学生   成绩 一班 ...

  5. tuple放入dict中

    tuple放入dict中是否可以正常运行 # 将tuple放入dict中 a = ('AI','Kobe','Yao') b = ('AI',['Kobe','Yao']) dict1 = {'a': ...

  6. 用MT.exe将exe中的manifest文件提取出来和将manifest文件放入exe中

     前一种方法是将manifest文件放入exe中,但是要记得需要在工程中设置 这样的话exe中就不存在manifest了,在debug目录下就会看到相应的manifest文件.后者是将exe中的man ...

  7. pyqt字符串分离开,放入列表中

    string1 = ''''' the stirng Has many line In THE fIle ''' list_of_string = string1.split() print list ...

  8. .Net中把图片等文件放入DLL中,并在程序中引用

    原文:.Net中把图片等文件放入DLL中,并在程序中引用 [摘要] 有时我们需要隐藏程序中的一些资源,比如游戏,过关后才能看到图片,那么图片就必须隐藏起来,否则不用玩这个游戏就可以看到你的图片了,呵呵 ...

  9. MyBatis 遍历数组放入in中

    必须要遍历出数组的值放入in中 如果直接将"'2','3','4','5','6','7','8'" 字符串放入in中,只会查出 inv_operate_type的值为2的数据,因 ...

随机推荐

  1. 第五章——定时器Timer序言

    定时器很重要. 上家公司有用的,是用来做定期数据同步的. 以前老同学有用到,曾经就定时器讨论过一次,还给过一次他我关于spring-task的总结. 但是并没有意识到定时器与多线程的关系,或者说,上一 ...

  2. intellij IDEA里各图标对应的文件类型

    本篇内容为大家提供的是IntelliJ IDEA 使用教程中的常见文件类型的图标介绍,IntelliJ IDEA是java语言开发的集成环境,IntelliJ在业界被公认为最好的java开发工具之一, ...

  3. java做成windows服务,电子秤例子,开机自动启动

    使用Java Service Wrapper工具制作 1.windows32位下载地址 https://sourceforge.net/projects/wrapper/files/ 2.window ...

  4. Cenos 6.5上的subverion的yum配置笔记

    Subversion在CenOS 6.5上的安装配置 1.安装 yum install subversion 2.配置   #创建目录   mkdir /opt/svn   #创建版本库   svna ...

  5. 配置SQL Server on Linux(2)

    1. 前言 前一篇配置SQL Server on Linux(1),地址:http://www.cnblogs.com/fishparadise/p/8125203.html ,是关于更改数据库排序规 ...

  6. 【CSS3】颜色表示方式

    <!DOCTYPE html> <html> <head> <meta charset="utf-8"> <title> ...

  7. Ubuntu字符界面输入密码始终提示错误 login incorrect 解决办法

    方法1. 先输入用户名按enter,然后ctrl+alt+f7 切换回图形界面,然后再切换到字符界面输密码 方法2.先输入用户名按enter,然后关闭小键盘上的numlock按键再打开,再输入密码 f ...

  8. 最长回文子序列(LPS)

    问题描述: 回文是正序与逆序相同的非空字符串,例如"civic"."racecar"都是回文串.任意单个字符的回文是其本身. 求最长回文子序列要求在给定的字符串 ...

  9. 命令行执行php脚本 中$argv和$argc

    在实际工作中有可能会碰到需要在nginx命令行执行php脚本的时候,当然你可以去配置一个conf用外网访问. 在nginx命令行中 使用 php index.php 就可以执行这个index.php脚 ...

  10. Python模块之pickle(列表,字典等复杂数据类型与二进制文件的转化)

    1.pickle模块简介 The pickle module implements binary protocols for serializing and de-serializing a Pyth ...