MyBatis批量操作报错:Parameter 'xxxList' not found. Available parameters are [list]

2018-03-27 13:59:52

问题背景:

在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>  

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

  • 2021-01-12 22:15:46

    NSLog各种打印占位符

    打印CGRect : NSLog(@"%@",NSStringFromCGRect(someCGRect)); 或者CFShow(NSStringFromCGRect(someCGRect));

  • 2021-01-13 13:36:29

    shortid nodejs短id生成器

    短ID在实际运用中很广泛, 其中比较典型的运用就是短地址。 市面上肯定有不少开源的生成短ID库, 基于node.js的估计也不少。 鉴于本人已然是node.js的脑残粉(本职java开发), 很多业余项目从前端到后端都基于javascript开发, 加上npm和bower的包管理以及grunt的打包工具, 在项目开发过程中体验特别酸爽。 由于当时项目前后端都会用到短ID, 但没找到合适的库同时支持npm和bower的(可能孤陋寡闻). 因此自己乐此不疲地又造了个轮子js-shortid(夷,为什么会说又呢?!). 下面主要介绍它的实现方案, 自认为比较优雅简洁。

  • 2021-01-13 17:23:21

    CREATE TABLE 表名 AS SELECT 语句 快速复制表但是锁表

    注意Table2的主键约束,如果Table2有主键而且不为空,则 field1, field2…中必须包括主键 在执行语句的时候,MySQL是逐行加锁的(扫描一个锁一个),直至锁住所有符合条件的数据,执行完毕才释放锁。所以当业务在进行的时候,切忌使用这种方法。 在RR隔离级别下,还会加行锁和间隙锁

  • 2021-01-13 17:27:04

    Navicat配置mysql数据库用户权限

    用数据库的时候就会遇到有多个用户,分配用户权限的情况,有些用户只读,有些用户可以读写,有些用户只能操作一个或者多个数据库,如何给mysql的用户设置权限,我这里描述一下如何用navicat图形操作分配用户权限

  • 2021-01-14 06:15:04

    通过glide获取图片显示后的真正宽高

    有时候需要获取网络图片的宽高来设置图片显示的大小,很多人会直接利用Glide的加载监听去拿图片的宽高,但是这样拿到的不是图片真正的宽高,而是图片显示在ImageView后的宽高。如下:

  • 2021-01-14 09:38:57

    Chrome插件详细教程

    严格来讲,我们正在说的东西应该叫Chrome扩展(Chrome Extension),真正意义上的Chrome插件是更底层的浏览器功能扩展,可能需要对浏览器源码有一定掌握才有能力去开发。鉴于Chrome插件的叫法已经习惯,本文也全部采用这种叫法,但读者需深知本文所描述的Chrome插件实际上指的是Chrome扩展。