zoukankan      html  css  js  c++  java
  • MyBatis批量操作报错:Parameter 'xxxList' not found. Available parameters are [list]

    问题背景:

    在Dao中使用MyBatis进行查询操作,参数是传的一个List:studentNameList,但是在执行查询的时候报错,具体日志如下:

    Shell代码  收藏代码
    1. com.chenzhou.base.mybatis.IbatisSystemException: SqlSession operation; nested exception is org.apache.ibatis.exceptions.PersistenceException:   
    2. ### Error querying database.  Cause: org.apache.ibatis.binding.BindingException: Parameter 'studentNameList' not found. Available parameters are [list]  
    3. ### Cause: org.apache.ibatis.binding.BindingException: Parameter 'studentNameList' not found. Available parameters are [list]  
    4.     at com.chenzhou.base.mybatis.SqlSessionTemplate.wrapException(SqlSessionTemplate.java:341)  
    5.     at com.chenzhou.base.mybatis.SqlSessionTemplate.execute(SqlSessionTemplate.java:127)  
    6.     at com.chenzhou.base.mybatis.SqlSessionTemplate.execute(SqlSessionTemplate.java:106)  
    7.     at com.chenzhou.base.mybatis.SqlSessionTemplate.selectOne(SqlSessionTemplate.java:138)  
    8.     at com.chenzhou.dao.GenericMybatisDao.count(GenericMybatisDao.java:306)  
    9.     at com.chenzhou.cds.ps.dao.impl.StudentDao.getStudentCount(StudentDao.java:42)  
    10.     at com.chenzhou.cds.ps.dao.impl.StudentDao$$FastClassByCGLIB$$8819e766.invoke(<generated>)  
    11.     at net.sf.cglib.proxy.MethodProxy.invoke(MethodProxy.java:191)  
    12.     at org.springframework.aop.framework.Cglib2AopProxy$CglibMethodInvocation.invokeJoinpoint(Cglib2AopProxy.java:689)  
    13.     at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:150)  
    14.     at org.springframework.aop.aspectj.MethodInvocationProceedingJoinPoint.proceed(MethodInvocationProceedingJoinPoint.java:80)  
    15.     at com.chenzhou.util.LogUtil.doMethodInfo(LogUtil.java:85)  
    16.     at com.chenzhou.util.LogUtil.doDebugMethodLog(LogUtil.java:36)  
    17.     at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)  
    18.     at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39)  
    19.     at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25)  
    20.     at java.lang.reflect.Method.invoke(Method.java:597)  
    21.     at org.springframework.aop.aspectj.AbstractAspectJAdvice.invokeAdviceMethodWithGivenArgs(AbstractAspectJAdvice.java:621)  
    22.     at org.springframework.aop.aspectj.AbstractAspectJAdvice.invokeAdviceMethod(AbstractAspectJAdvice.java:610)  
    23.     at org.springframework.aop.aspectj.AspectJAroundAdvice.invoke(AspectJAroundAdvice.java:65)  
    24.     at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:172)  
    25.     at org.springframework.aop.aspectj.AspectJAfterThrowingAdvice.invoke(AspectJAfterThrowingAdvice.java:55)  
    26.     at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:172)  
    27.     at org.springframework.transaction.interceptor.TransactionInterceptor.invoke(TransactionInterceptor.java:110)  
    28.     at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:172)  
    29.     at org.springframework.aop.interceptor.ExposeInvocationInterceptor.invoke(ExposeInvocationInterceptor.java:90)  
    30.     at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:172)  
    31.     at org.springframework.aop.framework.Cglib2AopProxy$DynamicAdvisedInterceptor.intercept(Cglib2AopProxy.java:622)  
    32.     at com.chenzhou.cds.ps.dao.impl.StudentDao$$EnhancerByCGLIB$$d4fcf513.getStudentCount(<generated>)  
    33.     at com.chenzhou.ps.dao.StudentDaoTest.testgetStudentCount(StudentDaoTest.java:44)  
    34. ……  

    单元测试用例代码如下:

    Java代码  收藏代码
    1. @Test  
    2. public void testgetStudentCount(){  
    3.     List<String> studentNameList = new ArrayList<String>();  
    4.     studentNameList.add("chenzhou");  
    5.     studentNameList.add("zhangsan");  
    6.     studentNameList.add("lisi");  
    7.     int count = studentDao.getStudentCount(studentNameList);  
    8.     System.out.println(count);  
    9. }  

    studentDao中的getStudentCount方法代码如下:

    Java代码  收藏代码
    1. public int getStudentCount(List<String> studentNameList){  
    2.     return super.count("getStudentCount", studentNameList);  
    3. }  

    MyBatis mapper.xml定义如下:

    Xml代码  收藏代码
    1. <!-- 查询学生数量  -->  
    2. <select id="Student.getStudentCount" parameterType="java.util.List" resultType="java.lang.Integer">  
    3.     <![CDATA[ 
    4.     SELECT 
    5.         COUNT(*) 
    6.     FROM 
    7.         t_student WHERE 1=1  
    8.     ]]>  
    9.     <if test="studentNameList != null">  
    10.         AND student_name in  
    11.         <foreach collection="studentNameList" item="item" open="(" separator="," close=")">  
    12.             #{item}   
    13.         </foreach>  
    14.     </if>  
    15. </select>  

    根据报错日志分析,是MyBatis在解析xml时找不到其中声明的studentNameList,但是在Dao中明明传的参数就是studentNameList,怎么会报错呢?

    查询了一下MyBatis官方的说明文档,终于找到了原因,在http://mybatis.github.io/mybatis-3/zh/dynamic-sql.html#foreach里有一段说明:

    写道
    注意 你可以传递一个 List 实例或者数组作为参数对象传给 MyBatis。当你这么做的时 候,MyBatis 会自动将它包装在一个 Map 中,用名称在作为键。List 实例将会以“list” 作为键,而数组实例将会以“array”作为键。

    因为我传的参数只有一个,而且传入的是一个List集合,所以mybatis会自动封装成Map<"list",studentNameList>。在解析的时候会通过“list”作为Map的key值去寻找。但是我在xml中却声明成studentNameList了,所以自然会报错找不到。

    解决办法:

    第一种就是修改mapper.xml中foreach标签内容,把studentNameList修改为list

    Xml代码  收藏代码
    1. <if test="list != null">  
    2.     AND student_name in  
    3.     <foreach collection="list" item="item" open="(" separator="," close=")">  
    4.         #{item}   
    5.     </foreach>  
    6. </if>  

    不过这种方式我个人不太建议,因为以后如果要扩展该方法,增加集合参数的时候,还得修改xml中的内容。

    第二种方式,修改dao中的参数传入方式,手动封装成map,然后把map当参数传进去

    Dao方法修改为:

    Java代码  收藏代码
    1. public int getStudentCount(List<String> studentNameList){  
    2.     //把参数手动封装在Map中  
    3.     Map<String, Object> map = new HashMap<String, Object>();  
    4.     map.put("studentNameList", studentNameList);  
    5.     return super.count("getStudentCount", map);  
    6. }  

    然后修改mapper.xml中的parameterType类型为Map

    Xml代码  收藏代码
    1. <!--注意下面的parameterType类型必须修改为Map类型,foreach中引用的List名称不用改变-->  
    2. <select id="Student.getStudentCount" parameterType="java.util.Map" resultType="java.lang.Integer">  
    3.     <![CDATA[ 
    4.     SELECT 
    5.         COUNT(*) 
    6.     FROM 
    7.         t_student WHERE 1=1  
    8.     ]]>  
    9.     <if test="studentNameList != null">  
    10.         AND student_name in  
    11.         <foreach collection="studentNameList" item="item" open="(" separator="," close=")">  
    12.             #{item}   
    13.         </foreach>  
    14.     </if>  
    15. </select>  

    修改完后,重新执行了一下测试用例,测试通过。

  • 相关阅读:
    真爱 vs. 种姓:新一代印度人的婚恋观
    美国司法部解禁guns打印技术
    特朗普访英,吃瓜群众却只想看《真爱至上》
    Semaphore(信号量)
    RLock(递归锁)
    用python编写九九乘法表
    php传值和传引用的区别
    post请求的header
    Content-type详解
    thinkphp5 学习笔记
  • 原文地址:https://www.cnblogs.com/winkey4986/p/3480328.html
Copyright © 2011-2022 走看看