hibernate总结三

在平时开发中Hibernate提供的hql基本能够满足我们的日常需求。但是在有些特殊的情况下,还是需要使用原生的sql,并且希望sql查询出来的结果能够绑定到pojo上。hibernate API中的createSQLQuery  和createQuery接口。
 
就像在这次的项目中,因为表结构要变化,有个新的需求:
要从一个表中查询极个别的字段并且还有几个是求多条数据的和的,之前一直在使用hibernate的API,使用的都是映射过得对象。
原对象及表结构是这样的:
 
  1. @Entity  
  2. @Table(name="T_BILL_ACCT_ITEM")  
  3. @NamedQuery(name="TBillAcctItem.findAll", query="SELECT t FROM TBillAcctItem t")  
  4. public class TBillAcctItem implements Serializable {  
  5.     private static final long serialVersionUID = 1L;  
  6.     @Id  
  7.     @Column(name="ACCT_ITEM_ID")  
  8.     private long acctItemId;  
  9.       
  10.     @Column(name="ACCT_ID")  
  11.     private long acctId;  
  12.       
  13.     @Column(name="BILLING_CYCLE_ID")  
  14.     private String billingCycleId;  
  15.       
  16.     @Column(name="PRODUCT_ID")  
  17.     private String productId;  
  18.       
  19.     @Column(name="SERVICE_ID")  
  20.     private String serviceId;  
  21.       
  22.     @Column(name="ACCT_ITEM_CODE")  
  23.     private String acctItemCode;  
  24.       
  25.     @Column(name="ORIGINAL_AMOUNT")  
  26.     private int originalAmount;  
  27.       
  28.     @Column(name="CDR_DISCOUNT")  
  29.     private int cdrDiscount;  
  30.       
  31.     @Column(name="ACCT_DISCOUNT")  
  32.     private int acctDiscount;  
  33.       
  34.     @Column(name="RECE_AMOUNT")  
  35.     private int receAmount;  
  36.       
  37.     @Column(name="REAL_AMOUNT")  
  38.     private int realAmount;  
  39.       
  40.     @Column(name="CHARGE_OFF_SOURCE")  
  41.     private int chargeOffSource;  
  42.       
  43.     @Column(name="STATE")  
  44.     private int state;  
但是我现在需要查询出来的信息是这样的一个pojo类:(因为int类型的值不能为空,所以我需要将字段定义为 java.long.Integer类型)
 
  1. public class ProductBillInfo implements Serializable{  
  2.   
  3.     private static final long serialVersionUID = 1L;  
  4.     private String productId;//产品标识  
  5.     private String serverId;//服务标示  
  6.     private Integer realAmount;//实收金额(sum求和)  
  7.     private Integer receAmount;//应收金额(sum求和)  
  8.     private int state;//账目状态  
  9.     private String billingCycle;//账期(月)  

因为hibernate要查询的对象是要和表结构一一映射的,现在求和字段这些个是映射不了的,
之前的方法就有问题了。 找了很多看了hibernate的API 找到这个东西。  如下:
 
 
 
 
大概的使用过程:
[java] view plain copy
 
 print?在CODE上查看代码片派生到我的代码片
  1.        StringBuffer b = new StringBuffer();    
  2. //... 省略SQL拼接代码    
  3. String sql = b.toString();    
  4. SQLQuery query = session.createSQLQuery(sql);   
  5. query.addScalar("productId", StandardBasicTypes.STRING);//要查询出来的字段、类型  
  6. List<ProductBillInfo> lnfo =query.list();  

下面是我用到的查询pojo的方法中的实现:
 
 Warnings("unchecked")  
  1.     public List<ProductBillInfo> findByAcctIdAndCycle1(long acctId,  
  2.             String cycleBegin, String cycleEnd, int state){  
  3.           
  4.         List<String> cycles = DateUtil.getAllCycle(cycleBegin, cycleEnd);  
  5.         StringBuffer buf = new StringBuffer();  
  6.         buf.append("select t.PRODUCT_ID as productId , t.SERVICE_ID as  serviceId ," +  
  7.         "sum(t.REAL_AMOUNT) as  realAmount  ,sum(t.RECE_AMOUNT) as  receAmount ," +  
  8.         "t.STATE as state ,t.BILLING_CYCLE_ID as  billingCycleId " +  
  9.         " from T_BILL_ACCT_ITEM t where t.STATE= '" + state + "' AND t.ACCT_ID='" + acctId + "' AND t.BILLING_CYCLE_ID in (");  
  10.         StringBuffer bf = new StringBuffer();  
  11.         for(String id : cycles){  
  12.                if(bf.toString().equals("")){  
  13.                    bf.append("'"+id+"'");  
  14.                } else{  
  15.                    bf.append(","+"'"+id+"'");  
  16.                }  
  17.         }  
  18.         buf.append(bf.toString());  
  19.         buf.append(") GROUP BY t.PRODUCT_ID,t.BILLING_CYCLE_ID,t.SERVICE_ID");  
  20.           
  21.         SQLQuery query = getSession().createSQLQuery(buf.toString());    
  22.         query.addScalar("productId", StandardBasicTypes.STRING);  
  23.         query.addScalar("serviceId", StandardBasicTypes.STRING);  
  24.         query.addScalar("realAmount", StandardBasicTypes.INTEGER);  
  25.         query.addScalar("receAmount", StandardBasicTypes.INTEGER);  
  26.         query.addScalar("state", StandardBasicTypes.INTEGER);  
  27.         query.addScalar("billingCycleId", StandardBasicTypes.STRING);  
  28.           
  29.         query.setResultTransformer(Transformers.aliasToBean(ProductBillInfo.class));    
  30.         List<ProductBillInfo> lnfo =query.list();  
  31.         return lnfo;  
  32.     }  
注:示例中addScalar及setResultTransformer很关键,我在使用的时候还报了一个错(原因是在实体中需要有一个无参构造函数),否则错误如下:
 
  1. 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(‘’、‘’、‘’)的语法,于是写了下面的一个拼接小方法:
 
  1. public static void main(String[] args) {  
  2.         List<String> c = new ArrayList<String>();  
  3.         c.add("1");c.add("2");c.add("3");  
  4.         StringBuffer bf = new StringBuffer();  
  5.         for(String id : c){  
  6.                if(bf.toString().equals("")){  
  7.                    bf.append("'"+id+"'");  
  8.                } else{  
  9.                    bf.append(","+"'"+id+"'");  
  10.                }  
  11.             }  
  12.         System.out.println(bf.toString());  
  13.     }  
打印结果是:     '1','2','3'    
使用StringBuffer拼接到原生的sql语句中
 
原文地址:http://blog.csdn.net/liwf_/article/details/39502745
posted @ 2016-03-29 15:53  FEI_>.<_JI  阅读(145)  评论(0编辑  收藏  举报