【问题标题】:Why CTE query not working in Hibernate?为什么 CTE 查询在 Hibernate 中不起作用?
【发布时间】:2017-06-20 12:24:21
【问题描述】:

这是我的 CTE 查询

String query = "WITH Search(id, name, merchantId, merchantName,merchantNo, purpose, maxDate, paymentAmount) AS (SELECT PAS.GATEWAY_ID AS id, (SELECT NAME FROM GATEWAY_DETAILS WHERE GATEWAY_ID=(SELECT GATEWAY_ID FROM GATEWAY WHERE GATEWAY_ID=PAS.GATEWAY_ID)) AS NAME, PD.merchantName_ID, (SELECT merchantName_NAME FROM merchantName_MASTER WHERE merchantName_ID=PD.merchantName_ID) AS merchantName, PD.TXN_NO, (SELECT MERCHANT_DESC FROM MERCHANTS WHERE DESC_ID=PAS.DESC_ID) AS PURPOSE, PD.TXN_DATE AS maxDate,CHEQUE_AMOUNT FROM TXN_DETAILS PD INNER JOIN GATEWAY_PASS PAS ON PD.PMNT_TXN_ID=PAS.PMNT_TXN_ID WHERE  PD.TXN_DATE  BETWEEN '2016/10/28' AND '2016/12/28' AND PD.GATEWAY_ID=PAS.GATEWAY_ID), MaxDate1(id1, MAX_DATE) AS (SELECT PD.GATEWAY_ID, MAX(maxDate) AS MAX_DATE FROM TXN_DETAILS PD, Search SMD WHERE GATEWAY_ID IN(SELECT id FROM Search) GROUP BY PD.GATEWAY_ID), MaxDate2(id1, MAX_DATE) AS ( SELECT PD.GATEWAY_ID, MAX(PD.TXN_DATE) FROM TXN_DETAILS PD, Search SMD WHERE PD.GATEWAY_ID IN(SMD.id) GROUP BY PD.GATEWAY_ID ), TxnDetails(GATEWAY1,  txnDate1, GATEWAY2, txnDate2) AS (  SELECT MD1.id1,MD2.id1,MD1.MAX_DATE, MD2.MAX_DATE FROM MaxDate1 MD1  INNER JOIN  MaxDate2 MD2 ON MD1.id1=MD2.id1 WHERE MD1.MAX_DATE=MD2.MAX_DATE ) SELECT PD.GATEWAY_ID, (SELECT NAME FROM GATEWAY_DETAILS    WHERE GATEWAY_ID=(SELECT GATEWAY_ID FROM GATEWAY WHERE GATEWAY_ID=PD.GATEWAY_ID)) AS NAME,  (SELECT merchantName_NAME FROM merchantName_MASTER WHERE merchantName_ID=PD.merchantName_ID) AS merchantName_NAME,  PD.TXN_NO, PD.TXN_DATE, PD.CHEQUE_AMOUNT,   (SELECT DESC_ID FROM MERCHANTS WHERE DESC_ID IN (SELECT DESC_ID FROM GATEWAY_PASS WHERE PMNT_TXN_ID=PD.PMNT_TXN_ID)) AS DESC_ID ,    (SELECT MERCHANT_DESC FROM MERCHANTS WHERE DESC_ID IN (SELECT DESC_ID FROM GATEWAY_PASS WHERE PMNT_TXN_ID=PD.PMNT_TXN_ID)) AS PURPOSE , PD.TXN_DATE FROM TXN_DETAILS PD WHERE PD.GATEWAY_ID IN (SELECT GATEWAY1 FROM TxnDetails) AND PD.TXN_DATE IN (SELECT txnDate2 FROM TxnDetails)";

这是 DAO

  List<Object[]> ObjList = entityManager.createNativeQuery(query).getResultList();
            List<TxntxnDTO> txnDtoList = new ArrayList();

            for (Object[] object : ObjList) {
                TxnDTO txnDTO = new TxnDTO();
                txnDTO.setid((String) object[0]);
                txnDTO.setname((String) object[1]);
                txnDTO.setmerchantNameName((String) object[2]);
                txnDTO.setmerchantNo((String) object[3]);
                txnDTO.setTxnDate((Date) object[4]);
                txnDTO.setTxnAmount((BigDecimal) object[5]);
                txnDTO.setMerchantId((Integer) object[6]);
                txnDTO.setTxnDesc((String) object[7]);

                txnDtoList.add(txnDTO);
            }

查询在 SQL 中运行良好,但在 DAO 中出现错误

javax.persistence.PersistenceException:org.hibernate.MappingException:没有 JDBC 类型的方言映射:-9

【问题讨论】:

  • 我建议,尝试在正常查询中更改它。
  • 嗨,红人,正常查询工作正常。但是我需要将一个查询的结果与另一个查询的结果进行比较,这就是我选择 CTE 的原因。
  • 你是否在休眠属性中指定了方言?提供数据源配置块
  • JDBC 类型 -9java.sql.Types.NVARCHAR。这似乎表明您的 Hibernate 方言不提供此类型的映射。确保您使用的是正确的方言(NVARCHAR 需要 SQLServer2008Dialect 或更高版本)

标签: java sql-server hibernate java-7 spring-4


【解决方案1】:

我已经通过关注解决了这个问题

这是我的 DTO 课程

    public class PaymentPostingDTO {

    private String id;

    private String name;

    private String merchantId;

    private String merchantName;

    private Integer merchantNo;

    private String purpose;

    private Date maxDate;

    private BigDecimal paymentAmount;

    /**
     * @return the id
     */
    public String getId() {
        return id;
    }

    /**
     * @param id
     *            the id to set
     */
    public void setId(String id) {
        this.id = id;
    }

    /**
     * @return the name
     */
    public String getName() {
        return name;
    }

    /**
     * @param name
     *            the name to set
     */
    public void setName(String name) {
        this.name = name;
    }

    /**
     * @return the merchantId
     */
    public String getMerchantId() {
        return merchantId;
    }

    /**
     * @param merchantId
     *            the merchantId to set
     */
    public void setMerchantId(String merchantId) {
        this.merchantId = merchantId;
    }

    /**
     * @return the merchantName
     */
    public String getMerchantName() {
        return merchantName;
    }

    /**
     * @param merchantName
     *            the merchantName to set
     */
    public void setMerchantName(String merchantName) {
        this.merchantName = merchantName;
    }

    /**
     * @return the merchantNo
     */
    public Integer getMerchantNo() {
        return merchantNo;
    }

    /**
     * @param merchantNo
     *            the merchantNo to set
     */
    public void setMerchantNo(Integer merchantNo) {
        this.merchantNo = merchantNo;
    }

    /**
     * @return the purpose
     */
    public String getPurpose() {
        return purpose;
    }

    /**
     * @param purpose
     *            the purpose to set
     */
    public void setPurpose(String purpose) {
        this.purpose = purpose;
    }

    /**
     * @return the maxDate
     */
    public Date getMaxDate() {
        return maxDate;
    }

    /**
     * @param maxDate
     *            the maxDate to set
     */
    public void setMaxDate(Date maxDate) {
        this.maxDate = maxDate;
    }

    /**
     * @return the paymentAmount
     */
    public BigDecimal getPaymentAmount() {
        return paymentAmount;
    }

    /**
     * @param paymentAmount
     *            the paymentAmount to set
     */
    public void setPaymentAmount(BigDecimal paymentAmount) {
        this.paymentAmount = paymentAmount;
    }

}

这是我的 DAO 实现

      Session session = getHibernateSession();
                String query = "WITH Search(id, name, merchantId, merchantName,merchantNo, purpose, maxDate, paymentAmount) AS 
        (
        SELECT PAS.GATEWAY_ID AS id, (SELECT NAME FROM GATEWAY_DETAILS WHERE GATEWAY_ID=(SELECT GATEWAY_ID FROM GATEWAY WHERE GATEWAY_ID=PAS.GATEWAY_ID)) AS NAME, PD.merchantName_ID, (SELECT merchantName_NAME FROM merchantName_MASTER WHERE merchantName_ID=PD.merchantName_ID) AS merchantName, PD.TXN_NO, (SELECT MERCHANT_DESC FROM MERCHANTS WHERE DESC_ID=PAS.DESC_ID) AS PURPOSE, PD.TXN_DATE AS maxDate,CHEQUE_AMOUNT FROM TXN_DETAILS PD INNER JOIN GATEWAY_PASS PAS ON PD.PMNT_TXN_ID=PAS.PMNT_TXN_ID WHERE  PD.TXN_DATE  BETWEEN '2016/10/28' AND '2016/12/28' AND PD.GATEWAY_ID=PAS.GATEWAY_ID), MaxDate1(id1, MAX_DATE) AS (SELECT PD.GATEWAY_ID, MAX(maxDate) AS MAX_DATE FROM TXN_DETAILS PD, Search SMD WHERE GATEWAY_ID IN(SELECT id FROM Search) GROUP BY PD.GATEWAY_ID), MaxDate2(id1, MAX_DATE) AS ( SELECT PD.GATEWAY_ID, MAX(PD.TXN_DATE) FROM TXN_DETAILS PD, Search SMD WHERE PD.GATEWAY_ID IN(SMD.id) GROUP BY PD.GATEWAY_ID ), TxnDetails(GATEWAY1,  txnDate1, GATEWAY2, txnDate2) AS (  SELECT MD1.id1,MD2.id1,MD1.MAX_DATE, MD2.MAX_DATE FROM MaxDate1 MD1  INNER JOIN  MaxDate2 MD2 ON MD1.id1=MD2.id1 WHERE MD1.MAX_DATE=MD2.MAX_DATE ) SELECT PD.GATEWAY_ID, (SELECT NAME FROM GATEWAY_DETAILS    WHERE GATEWAY_ID=(SELECT GATEWAY_ID FROM GATEWAY WHERE GATEWAY_ID=PD.GATEWAY_ID)) AS NAME,  (SELECT merchantName_NAME FROM merchantName_MASTER WHERE merchantName_ID=PD.merchantName_ID) AS merchantName_NAME,  PD.TXN_NO, PD.TXN_DATE, PD.CHEQUE_AMOUNT,   (SELECT DESC_ID FROM MERCHANTS WHERE DESC_ID IN (SELECT DESC_ID FROM GATEWAY_PASS WHERE PMNT_TXN_ID=PD.PMNT_TXN_ID)) AS DESC_ID ,    (SELECT MERCHANT_DESC FROM MERCHANTS WHERE DESC_ID IN (SELECT DESC_ID FROM GATEWAY_PASS WHERE PMNT_TXN_ID=PD.PMNT_TXN_ID)) AS PURPOSE , PD.TXN_DATE FROM TXN_DETAILS PD WHERE PD.GATEWAY_ID IN (SELECT GATEWAY1 FROM TxnDetails) AND PD.TXN_DATE IN (SELECT txnDate2 FROM TxnDetails
        )";

        SQLQuery sqlQuery = session.createSQLQuery(query);
        sqlQuery.addScalar("id", StringType.INSTANCE);
        sqlQuery.addScalar("name", StringType.INSTANCE);
        sqlQuery.addScalar("merchantId", StringType.INSTANCE);
        sqlQuery.addScalar("merchantName", StringType.INSTANCE);
        sqlQuery.addScalar("merchantNo", IntegerType.INSTANCE);
        sqlQuery.addScalar("purpose", StringType.INSTANCE);
        sqlQuery.addScalar("maxDate", DateType.INSTANCE);
        sqlQuery.addScalar("paymentAmount", BigDecimalType.INSTANCE);
  sqlQuery.setResultTransformer(Transformers.aliasToBean(PaymentPostingDTO.class));
    List<PaymentPostingDTO> paymentList = sqlQuery.list();

【讨论】:

    【解决方案2】:

    我设法通过转换返回对象来解决这个问题:

    SELECT CAST(field AS VARCHAR(20)) AS field
    

    https://stackoverflow.com/a/12986024/2454665 看来,有时数据库会以无法映射到 Hibernate 类型的奇怪类型返回自定义 SQL 查询的结果,因此是 CAST。

    【讨论】:

    • 谢谢@Nicolas。您的解决方案运行良好,但我通过添加 sqlQuery.addScalar("paymentAmount", BigDecimalType.INSTANCE); 得到了解决方法;
    • -9 不是“奇怪的类型”,它是Types.NVARCHAR
    • 嗨,马克,在那段时间我已经搜索过,但找不到任何解决该问题的方法。
    猜你喜欢
    • 1970-01-01
    • 2012-03-26
    • 1970-01-01
    • 1970-01-01
    • 2012-08-25
    • 2013-12-03
    • 1970-01-01
    • 2014-03-22
    • 2021-12-17
    相关资源
    最近更新 更多