无效的列类型 ~ 将 ArrayList<String> 发送到 pl/sql createdNameQuery

Posted

技术标签:

【中文标题】无效的列类型 ~ 将 ArrayList<String> 发送到 pl/sql createdNameQuery【英文标题】:invalid column type ~ Sending ArrayList<String> to a pl/sql createdNameQuery 【发布时间】:2011-08-01 21:39:31 【问题描述】:

这与我过去的 question 有关。

我收到List&lt;Employee&gt; 并从Employee 对象中获取ID 并将它们放入ArrayList&lt;String&gt;,然后将ArrayList 作为createdNameQuery 中的参数发送。我收到一个无效的列类型 sql 异常。我在 pl/sql developer 中测试了查询并返回了字段。我已经尝试通过在每个 id 之间放置一个逗号并发送它来构建一串 id,但是,我从该尝试中收到了一个异常。我很好奇我的查询设置不正确或发送数据不正确。

我的存储库中的函数:

public List<RequestByRequester> getRequestsByRequesters(
        List<Employee> employeeList) 
        throws NoDataFoundException 

    List<String> idList = new ArrayList<String>();
    for(Employee emp : employeeList)           
        idList.add(emp.getId().toString());
    

    log.debug("Input params[requesters=" + idList + "]");

    List<RequestByRequester> resultList = getEm().createNamedQuery(
            "requestByRequestor.getRequestsByRequesters", RequestByRequester.class)
            .setParameter(1,idList)
            .getResultList();

        if(resultList == null || resultList.size() <= 0)
            throw new NoDataFoundException("No requests found by requesters.");
        else
            return resultList;

我的命名查询requestByRequestor.getRequestsByRequesters如下:

   @NamedNativeQuery(
    name = "requestByRequestor.getRequestsByRequesters",
    resultClass = RequestByRequester.class,
    query = "SELECT EMP.EMPL_FIRST_NAME || ' ' || EMP.EMPL_LAST_NAME REQUESTER," +
            "       R.RQST_ID RQST_ID," +
            "       R.TITLE TITLE," +
            "       R.DESCRIPTION DESCRIPTION," +
            "       DECODE(R.RESOLUTION_DATE, NULL, 'Open', 'Closed') STATUS" +
            "  FROM TARTS.REQUESTS R, SYS_EMPLOYEES EMP" +
            " WHERE R.EMPL_ID_REQUESTED_BY = EMP.EMPL_ID" +
            "   AND EMP.EMPL_ID IN (?)" +
            " ORDER BY 1, 5 DESC, 2"
    )

编辑:按要求添加例外。

当我在查询中使用:ids 时出现此异常:

Internal Exception: java.sql.SQLException: Missing IN or OUT parameter at index:: 1 Error Code: 17041 Call: SELECT EMP.EMPL_FIRST_NAME || ' ' || EMP.EMPL_LAST_NAME REQUESTER, R.RQST_ID RQST_ID, R.TITLE TITLE, R.DESCRIPTION DESCRIPTION, DECODE(R.RESOLUTION_DATE, NULL, 'Open', 'Closed') STATUS FROM TARTS.REQUESTS R, SYS_EMPLOYEES EMP WHERE R.EMPL_ID_REQUESTED_BY = EMP.EMPL_ID AND EMP.EMPL_ID IN :ids ORDER BY 1, 5 DESC, 2 Query: ReadAllQuery(name="requestByRequestor.getRequestsByRequesters" referenceClass=RequestByRequester sql="SELECT EMP.EMPL_FIRST_NAME || ' ' || EMP.EMPL_LAST_NAME REQUESTER, R.RQST_ID RQST_ID, R.TITLE TITLE, R.DESCRIPTION DESCRIPTION, DECODE(R.RESOLUTION_DATE, NULL, 'Open', 'Closed') STATUS FROM TARTS.REQUESTS R, SYS_EMPLOYEES EMP WHERE R.EMPL_ID_REQUESTED_BY = EMP.EMPL_ID AND EMP.EMPL_ID IN :ids ORDER BY 1, 5 DESC, 2")

当我在查询中使用?1(?) 时出现异常:

Internal Exception: java.sql.SQLException: Invalid column type Error Code: 17004 Call: SELECT EMP.EMPL_FIRST_NAME || ' ' || EMP.EMPL_LAST_NAME REQUESTER, R.RQST_ID RQST_ID, R.TITLE TITLE, R.DESCRIPTION DESCRIPTION, DECODE(R.RESOLUTION_DATE, NULL, 'Open', 'Closed') STATUS FROM TARTS.REQUESTS R, SYS_EMPLOYEES EMP WHERE R.EMPL_ID_REQUESTED_BY = EMP.EMPL_ID AND EMP.EMPL_ID IN ? ORDER BY 1, 5 DESC, 2 bind => [[2192, 632]] Query: ReadAllQuery(name="requestByRequestor.getRequestsByRequesters" referenceClass=RequestByRequester sql="SELECT EMP.EMPL_FIRST_NAME || ' ' || EMP.EMPL_LAST_NAME REQUESTER, R.RQST_ID RQST_ID, R.TITLE TITLE, R.DESCRIPTION DESCRIPTION, DECODE(R.RESOLUTION_DATE, NULL, 'Open', 'Closed') STATUS FROM TARTS.REQUESTS R, SYS_EMPLOYEES EMP WHERE R.EMPL_ID_REQUESTED_BY = EMP.EMPL_ID AND EMP.EMPL_ID IN ? ORDER BY 1, 5 DESC, 2")

【问题讨论】:

那些 ORDER BY 术语应该是什么意思?它们不是 result_variable,也不是 state_field_path_expression。建议阅读 JPA 规范第 4.9 节。如果要按顺序使用结果表达式,请使用别名“SELECT something AS p”,然后可以按顺序引用“p”。相反,如果您的意思是 SQL,那么您使用的是错误的查询生成方法 请包含指定的查询定义 @James 它包含在文本下方 requestByRequestor.getRequestsByRequesters 你能发布你看到的异常吗? 我已经能够通过运行一个for每个循环来完成这个工作,该循环遍历idList一次运行1个id的查询并将结果设置为tempList然后.addAlltempList到主要结果列表。这不是一种干净的方法,而是一种创可贴的解决方案,直到有人能找出问题所在。 【参考方案1】:

问题(我认为,因为您没有包含命名查询的定义)是您正在执行本机 SQL 查询,而 List 不是有效的 SQL/JDBC 参数值。

EclipseLink 支持 JPQL 查询的列表参数,但不支持原生 SQL 查询。

您需要使用 JPQL,或者在 SQL 中定义每个参数

EMP.EMPL_ID IN (:id1, :id2, :id3)

.setParameter("id1", idList.get(0));
.setParameter("id2", idList.get(1));

【讨论】:

就像我在回复您之前的评论时提到的那样。我的命名查询在我的原始帖子中。请查看例外列表的正上方。它就在 EDIT:Adding exceptions as requested 行的正上方 我认为你是对的。将不得不使用 JPQL 查询。谢谢!【参考方案2】:

我认为问题出在这里

AND EMP.EMPL_ID IN (?)

表示位置参数,需要在问号后面加上数字。

 AND EMP.EMPL_ID IN (?1)

或者,你可以这样做

 AND EMP.EMPL_ID IN (:empIds) 

然后像这样调用您的查询

List<RequestByRequester> resultList = getEm().createNamedQuery(
        "requestByRequestor.getRequestsByRequesters", RequestByRequester.class)
        .setParameter("empIds", idList)
        .getResultList();

【讨论】:

我尝试了AND EMP.EMPL_ID IN (?1) 方法并收到无效列错误。当我尝试AND EMP.EMPL_ID IN (:empIds) 方法时,我收到了这个Internal Exception: java.sql.SQLException: Missing IN or OUT parameter at index 错误。【参考方案3】:

在您的代码中,您将 emp.getId() 映射到 String

List<String> idList = new ArrayList<String>();
    for(Employee emp : employeeList)           
        idList.add(emp.getId().toString());
    

但我敢打赌,此列 EMP.EMPL_ID 的类型是数字或类似的东西。当你执行

.setParameter("empIds", idList)

JPA 会将其转换为带有引号和转义字符等的字符串值,因此:

java.sql.SQLException:无效的列类型

尝试映射到List&lt;Long&gt; 或其他数字类型。

【讨论】:

EMP.EMPL_ID 是 Long,我尝试了 List,并根据之前的建议更改了查询,结果与以前相同。 您是使用 SqlResultSetMapping 还是使用 NamedQuery (requestByRequestor.getRequestsByRequesters) 指定 ResultClass? 是具体的ResultClass RequestByRequester 您使用的是 Hibernate、OpenJPA 还是其他一些 JPA 实现?【参考方案4】:

查询:

List<RequestByRequester> resultList = getEm().createNamedQuery(
    "requestByRequestor.getRequestsByRequesters", RequestByRequester.class)
    .setParameter("ids", idList)
    .getResultList();

NamedQuery WHERE 部分:

WHERE R.EMPL_ID_REQUESTED_BY = EMP.EMPL_ID
AND EMP.EMPL_ID IN :ids

注意:ids 没有括号。这应该可以。

【讨论】:

我试过不带括号,收到Internal Exception: java.sql.SQLException: Missing IN or OUT parameter at index

以上是关于无效的列类型 ~ 将 ArrayList<String> 发送到 pl/sql createdNameQuery的主要内容,如果未能解决你的问题,请参考以下文章

Spring JdbcTemplate 查询参数类型错误:无效的列类型

Wpf DataGrid插入列,列的类型问题

mybatis插入数据传入值为null时提示无效的列类型

csv导入错误的Mysql排序规则/类型第1362行的CSV输入中的列计数无效

框架中无效的列类型异常分析

如何使具有数据类型不匹配的列无效