问题背景:
在Dao中使用MyBatis进行查询操作,参数是传的一个List:studentNameList,但是在执行查询的时候报错,具体日志如下:
com.chenzhou.base.mybatis.IbatisSystemException: SqlSession operation; nested exception is org.apache.ibatis.exceptions.PersistenceException: ### Error querying database. Cause: org.apache.ibatis.binding.BindingException: Parameter 'studentNameList' not found. Available parameters are [list] ### Cause: org.apache.ibatis.binding.BindingException: Parameter 'studentNameList' not found. Available parameters are [list] at com.chenzhou.base.mybatis.SqlSessionTemplate.wrapException(SqlSessionTemplate.java:341) at com.chenzhou.base.mybatis.SqlSessionTemplate.execute(SqlSessionTemplate.java:127) at com.chenzhou.base.mybatis.SqlSessionTemplate.execute(SqlSessionTemplate.java:106) at com.chenzhou.base.mybatis.SqlSessionTemplate.selectOne(SqlSessionTemplate.java:138) at com.chenzhou.dao.GenericMybatisDao.count(GenericMybatisDao.java:306) at com.chenzhou.cds.ps.dao.impl.StudentDao.getStudentCount(StudentDao.java:42) at com.chenzhou.cds.ps.dao.impl.StudentDao$$FastClassByCGLIB$$8819e766.invoke(<generated>) at net.sf.cglib.proxy.MethodProxy.invoke(MethodProxy.java:191) at org.springframework.aop.framework.Cglib2AopProxy$CglibMethodInvocation.invokeJoinpoint(Cglib2AopProxy.java:689) at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:150) at org.springframework.aop.aspectj.MethodInvocationProceedingJoinPoint.proceed(MethodInvocationProceedingJoinPoint.java:80) at com.chenzhou.util.LogUtil.doMethodInfo(LogUtil.java:85) at com.chenzhou.util.LogUtil.doDebugMethodLog(LogUtil.java:36) at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method) at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39) at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25) at java.lang.reflect.Method.invoke(Method.java:597) at org.springframework.aop.aspectj.AbstractAspectJAdvice.invokeAdviceMethodWithGivenArgs(AbstractAspectJAdvice.java:621) at org.springframework.aop.aspectj.AbstractAspectJAdvice.invokeAdviceMethod(AbstractAspectJAdvice.java:610) at org.springframework.aop.aspectj.AspectJAroundAdvice.invoke(AspectJAroundAdvice.java:65) at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:172) at org.springframework.aop.aspectj.AspectJAfterThrowingAdvice.invoke(AspectJAfterThrowingAdvice.java:55) at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:172) at org.springframework.transaction.interceptor.TransactionInterceptor.invoke(TransactionInterceptor.java:110) at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:172) at org.springframework.aop.interceptor.ExposeInvocationInterceptor.invoke(ExposeInvocationInterceptor.java:90) at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:172) at org.springframework.aop.framework.Cglib2AopProxy$DynamicAdvisedInterceptor.intercept(Cglib2AopProxy.java:622) at com.chenzhou.cds.ps.dao.impl.StudentDao$$EnhancerByCGLIB$$d4fcf513.getStudentCount(<generated>) at com.chenzhou.ps.dao.StudentDaoTest.testgetStudentCount(StudentDaoTest.java:44) ……
单元测试用例代码如下:
@Test public void testgetStudentCount(){ List<String> studentNameList = new ArrayList<String>(); studentNameList.add("chenzhou"); studentNameList.add("zhangsan"); studentNameList.add("lisi"); int count = studentDao.getStudentCount(studentNameList); System.out.println(count); }
studentDao中的getStudentCount方法代码如下:
public int getStudentCount(List<String> studentNameList){ return super.count("getStudentCount", studentNameList); }
MyBatis mapper.xml定义如下:
<!-- 查询学生数量 --> <select id="Student.getStudentCount" parameterType="java.util.List" resultType="java.lang.Integer"> <![CDATA[ SELECT COUNT(*) FROM t_student WHERE 1=1 ]]> <if test="studentNameList != null"> AND student_name in <foreach collection="studentNameList" item="item" open="(" separator="," close=")"> #{item} </foreach> </if> </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
<if test="list != null"> AND student_name in <foreach collection="list" item="item" open="(" separator="," close=")"> #{item} </foreach> </if>
不过这种方式我个人不太建议,因为以后如果要扩展该方法,增加集合参数的时候,还得修改xml中的内容。
第二种方式,修改dao中的参数传入方式,手动封装成map,然后把map当参数传进去
Dao方法修改为:
public int getStudentCount(List<String> studentNameList){ //把参数手动封装在Map中 Map<String, Object> map = new HashMap<String, Object>(); map.put("studentNameList", studentNameList); return super.count("getStudentCount", map); }
然后修改mapper.xml中的parameterType类型为Map
<!--注意下面的parameterType类型必须修改为Map类型,foreach中引用的List名称不用改变--> <select id="Student.getStudentCount" parameterType="java.util.Map" resultType="java.lang.Integer"> <![CDATA[ SELECT COUNT(*) FROM t_student WHERE 1=1 ]]> <if test="studentNameList != null"> AND student_name in <foreach collection="studentNameList" item="item" open="(" separator="," close=")"> #{item} </foreach> </if> </select>
修改完后,重新执行了一下测试用例,测试通过。
相关推荐
>Window->prefenrence->XML->XML Catalog->User Specifiled Entreis->Add->Location(此处是你放dtd文件的位置例如:D:\mybatis\mybatis-3-config.dtd)->Key(如果更改config,此处应该是:-//mybatis.org//DTD ...
在整合Spring Boot与MyBatis-Plus的过程中,可能会遇到启动应用时报错的问题。错误信息通常类似于"Error starting ApplicationContext",并提示"UnsatisfiedDependencyException",这表明Spring框架在初始化bean时...
在使用Mybatis框架时,你可能会遇到`org.apache.ibatis.exceptions.PersistenceException`这样的异常。这个异常通常表示在执行数据库查询操作时遇到了问题。本篇将详细分析这个问题并提供解决方法。 ### 问题概述 ...
然而,当你遇到“_frch_item_0 not found”这样的错误时,通常是由于在使用`<foreach>`时出现了配置或编码上的问题。下面我们将深入探讨这个问题及其解决方法。 首先,我们要理解这个错误通常表示MyBatis在尝试访问...
首先,`Invalid bound statement (not found)` 错误通常出现在你尝试执行一个Mybatis-Plus的CRUD操作(如insert、update、delete或select)时,但Mybatis-Plus找不到对应的Mapper方法。这可能是由于以下原因: 1. *...
mybatis批量添加的时候报错总结报错 parameter'_frch_item_0 not found
IDEA离线安装MybatisX插件(MybatisX-1.1.2至1.4.17),...mybatis.xml,mapper.xml 提示 mapper 和 xml 支持类似 jpa 的自动提示(参考 MybatisCodeHelperPro) 集成 mybatis 生成器 Gui(来自免费的 mybatis 插件)
赠送jar包:mybatis-paginator-1.2.15.jar; 赠送原API文档:mybatis-paginator-1.2.15-javadoc.jar; 赠送源代码:mybatis-paginator-1.2.15-sources.jar; 赠送Maven依赖信息文件:mybatis-paginator-1.2.15.pom;...
IDEA离线安装MybatisX-1.4.x插件(MybatisX-1.4.10、...mybatis.xml,mapper.xml 提示 mapper 和 xml 支持类似 jpa 的自动提示(参考 MybatisCodeHelperPro) 集成 mybatis 生成器 Gui(来自免费的 mybatis 插件)
IDEA离线安装MybatisX-1.5.x插件(MybatisX-1.5.0、Mybatis...mybatis.xml,mapper.xml 提示 mapper 和 xml 支持类似 jpa 的自动提示(参考 MybatisCodeHelperPro) 集成 mybatis 生成器 Gui(来自免费的 mybatis 插件)
springmvc集成mybatis mybatis实战教程之六:与SpringMVC的集成.pdf
Mybatis-ehcache 1.2.1 是一个集成Mybatis和Ehcache的缓存模块,用于提高Mybatis框架的查询效率。Ehcache是一个广泛使用的Java缓存解决方案,它能够有效地存储和检索数据,减少数据库的负载,提高应用程序性能。在...
idea创建Maven项目时,报错显示Failed to execute goal org.apache.maven.plugins:maven-resources-plugin:3.0.2:resources,并且Maven插件内看不到 mybatis-generator。如下图: 折腾了好久发现配置放错地方了,...
在使用Mybatis进行数据库操作时,批量更新是一个常见的需求,特别是在处理大量数据的时候。然而,在实际操作中,可能会遇到批量更新报错的情况。本文将详细分析Mybatis批量更新的报错问题及其解决方法。 首先,报错...
1. `mybatis.jar`:这是MyBatis的核心库,包含了MyBatis的主要功能,如SQL映射文件解析、动态SQL生成等。它提供了SqlSessionFactoryBuilder、SqlSessionFactory和SqlSession等关键接口,是整个MyBatis框架的基础。 ...
mybatis-plus最新代码生成器项目源码 :mybatis-plus-generator.zip mybatis-plus最新代码生成器项目源码 :mybatis-plus-generator.zip mybatis-plus最新代码生成器项目源码 :mybatis-plus-generator.zip ...
mybatis 技巧:MyBatis 中的trim标签,好用!.zip技巧:MyBatis 中的trim标签,好用!.zip技巧:MyBatis 中的trim标签,好用!.zip技巧:MyBatis 中的trim标签,好用!.zip技巧:MyBatis 中的trim标签,好用!.zip...
在Java编程中,某些方法可能在特定对象或特定条件下不支持某个操作,当尝试执行这些操作时,就会抛出此异常。 在给定的描述中,问题出在MyBatis框架的使用上。MyBatis是一个优秀的持久层框架,它允许开发者将SQL...
### MySQL 数据库在 Spring Boot 项目中启动报错问题解析 #### 错误信息解读 在给定的信息中,我们注意到一个具体的错误信息:“com.baomidou.mybatisplus.core.exceptions.MybatisPlusException: Error:...