MyBatis错误

说明:记录一次MyBatis错误,错误信息如下,说数字转换异常,显然,把一个字符串类型转为数字类型,肯定是不行的。

text 复制代码
2024-08-29 19:44:43.198 ERROR 24216 --- [nio-9090-exec-2] o.a.c.c.C.[.[.[/].[dispatcherServlet]    : Servlet.service() for servlet [dispatcherServlet] in context with path [] threw exception [Request processing failed; nested exception is org.mybatis.spring.MyBatisSystemException: nested exception is org.apache.ibatis.exceptions.PersistenceException: 
### Error querying database.  Cause: java.lang.NumberFormatException: For input string: "hezy"
### Cause: java.lang.NumberFormatException: For input string: "hezy"] with root cause

java.lang.NumberFormatException: For input string: "hezy"

场景

(前端)

(接口)

java 复制代码
    @GetMapping("/test")
    public void test() {
        User user = new User();
        user.setId(1);
        user.setUsername("hezy");
        userRepository.selectUserById(user);
    }

(Repository)

java 复制代码
    User selectUserById(@Param("user") User user);

定位到项目的动态SQL,如下,这是我本地复现的Demo,在实际的项目中,是一个查询条件字段,表示某个分组,可选"所有分组"或者"具体某个分组",当时的开发应该是和前端约定过,当值为0时表示为所有分组,所以这个字段传递可能是0,或者是具体的分组ID,因此就有了类似下面这样的判断

(动态SQL)

xml 复制代码
    <select id="selectUserById" resultType="com.hezy.pojo.User">
        SELECT * FROM i_users WHERE id = #{user.id}
        <if test="user.username != null and user.username != '' and user.username != '0'">
            and username = #{user.username}
        </if>
        <if test="user.username == '0'">
            and username = '0'
        </if>
    </select>

分析&解决

显然,这里是有隐型的类型转换,但一眼看过去,动态SQL里面的判断都是字符串,0也用单引号包起来了。无计可施之际,我想到单引号在Java中表示的是字符,而字符又能自动转为数字进行计算,如下,所以我想是不是MyBatis底层把'0'当做数字了,然后把左边的参数也尝试转为数值类型,造成了这次错误。

于是,我想到把0通过双引号包起来,这铁定就是字符串了,双引号在XML里面需要通过&quot;转义,如下:

xml 复制代码
    <select id="selectUserById" resultType="com.hezy.pojo.User">
        SELECT * FROM i_users WHERE id = #{user.id}
        <if test="user.username != null and user.username != '' and user.username != &quot;0&quot;">
            and username = #{user.username}
        </if>
        <if test="user.username == &quot;0&quot;">
            and username = '0'
        </if>
    </select>

成功了,golder~,这个问题困扰了很久

解决 :if标签里面对数值的判断,如果你想要的是判断字符串,右边的数值需要用&quot;包起来。

相关推荐
皮皮林5513 小时前
SpringBoot 全局/局部双模式 Gzip 压缩实战:14MB GeoJSON 秒变 3MB
java·spring boot
weixin_456904273 小时前
Spring Boot 用户管理系统
java·spring boot·后端
趁你还年轻_3 小时前
异步编程CompletionService
java
DKPT3 小时前
Java内存区域与内存溢出
java·开发语言·jvm·笔记·学习
sibylyue3 小时前
Guava中常用的工具类
java·guava
奔跑吧邓邓子3 小时前
【Java实战㉞】从0到1:Spring Boot Web开发与接口设计实战
java·spring boot·实战·web开发·接口设计
专注API从业者3 小时前
Python/Java 代码示例:手把手教程调用 1688 API 获取商品详情实时数据
java·linux·数据库·python
奔跑吧邓邓子4 小时前
【Java实战㉝】Spring Boot实战:从入门到自动配置的进阶之路
java·spring boot·实战·自动配置
ONLYOFFICE4 小时前
【技术教程】如何将ONLYOFFICE文档集成到使用Spring Boot框架编写的Java Web应用程序中
java·spring boot·编辑器
叫我阿柒啊4 小时前
Java全栈开发工程师的实战面试经历:从基础到微服务
java·微服务·typescript·vue·springboot·前端开发·后端开发