iis服务器助手广告广告
返回顶部
首页 > 资讯 > 精选 >Mybatis查询语句条件为枚举类型时报错怎么办
  • 827
分享到

Mybatis查询语句条件为枚举类型时报错怎么办

2023-06-26 20:06:29 827人浏览 安东尼
摘要

这篇文章主要介绍mybatis查询语句条件为枚举类型时报错怎么办,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!Mybatis查询语句条件为枚举类型报错通常我们对于数据库中一些枚举字段使用tinyInt类型,而java

这篇文章主要介绍mybatis查询语句条件为枚举类型时报错怎么办,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!

Mybatis查询语句条件为枚举类型报错

通常我们对于数据库中一些枚举字段使用tinyInt类型,而java对象对应的字段很多时候会为了方便定义成short或者int。但这样显然不美观方便,让后面维护的人抠破脑袋找你的常量定义在哪儿,要是没有注释简直让人崩溃。时间久后,没有人知道这里面的值。只能一行行读源码

优雅的程序员当然想到了优雅的枚举,而mybatis“强大”的枚举类型处理器EnumOrdinalTypeHandler相信都不陌生。

然而配置枚举处理器花了九牛二虎之力改好原来的mapper运行测试用例全在报错。而插入、部分查询却没报错。这时进程进行到一半让人崩溃想要放弃。

通常这个错误是

"failed to invoke constructor for handler class org.apache.ibatis.type.EnumOrdinalTypeHandler”

原因是因为该死的查询条件使用枚举对象作为条件,无论你用selectExample还是其他的select,当条件where enum = #{enum}时就会报错。不要怀疑自己是不是EnumOrdinalTypeHandler没配对,如果没配对那一定会是所有的查询接口都会报错。

stackoverflow上只有一条相关问题。为什么这么少?这不是很常见的错误吗?jpa或hibernate就能很优雅的使用枚举啊。原因嘛,老外们很少用半自动的mybatis框架。只有国内奉为圭臬,原因嘛当然是听说人家阿里就用mybatis,所以一定是好的。也不看自己的业务到底是否真正触及到要提升sql性能的地步。

话说回来,目前给出来的答案似乎是mybatis的bug,但对于mybatis这种半自动框架这不一定是bug。

解决办法很简单粗暴,把where enum = #{enum}条件换成where enum in (***)万事大吉。但熟悉的同学已经发现了。这样的性能显然不如=。用short和int的同学肯定又开心了。看吧我就说数据库什么类型就用什么类型,枚举就是垃圾。说这话的同学显然还不习惯封装、规范这一套,更喜欢随心所欲的感觉。

今天的教训就到这。

Mybatis处理枚举类型

1、枚举

package com.ahut.core.enums;import java.util.HashMap;import java.util.Map;public enum SexEnum {    MAN("1", "男"), WOMAN("2", "女");    private String key;    private String value;    private static Map<String, SexEnum> sexEnumMap = new HashMap<>();    static {        for (SexEnum sexEnum : SexEnum.values()) {            sexEnumMap.put(sexEnum.geTKEy(), sexEnum);        }    }        private SexEnum(String key, String value) {        this.key = key;        this.value = value;    }        public static SexEnum getSexEnumByKey(String key) {        return sexEnumMap.get(key);    }    public String getKey() {        return key;    }    public void setKey(String key) {        this.key = key;    }    public String getValue() {        return value;    }    public void setValue(String value) {        this.value = value;    }}

2、包含枚举的实体类

package com.ahut.entity;import java.io.Serializable;import java.util.Date;import com.ahut.core.enums.SexEnum;public class Demo implements Serializable {        private static final long serialVersionUID = 4122974131420281791L;    private Date birthDay;    private String userName;    private int age;    private String id;    private SexEnum sex;    public Demo() {        super();        // TODO Auto-generated constructor stub    }    @Override    public String toString() {        return "Demo [id=" + id + ", userName=" + userName + ", age=" + age + ", birthDay=" + birthDay + ", sex=" + sex                + "]";    }    public String getId() {        return id;    }    public void setId(String id) {        this.id = id;    }    public String getUserName() {        return userName;    }    public void setUserName(String userName) {        this.userName = userName;    }    public int getAge() {        return age;    }    public void setAge(int age) {        this.age = age;    }    public Date getBirthDay() {        return birthDay;    }    public void setBirthDay(Date birthDay) {        this.birthDay = birthDay;    }    public SexEnum getSex() {        return sex;    }    public void setSex(SexEnum sex) {        this.sex = sex;    }}

3、书写枚举处理器

package com.ahut.handler;import java.sql.CallableStatement;import java.sql.PreparedStatement;import java.sql.ResultSet;import java.sql.SQLException;import org.apache.ibatis.type.BaseTypeHandler;import org.apache.ibatis.type.JdbcType;import com.ahut.core.enums.SexEnum;public class SexEnumHandler extends BaseTypeHandler<SexEnum> {        @Override    public void setNonNullParameter(PreparedStatement ps, int i, SexEnum parameter, JdbcType jdbcType)            throws SQLException {        // baseTypeHandler已经帮我们做了parameter的null判断        // 第二个参数 : 存入到数据库中的值        ps.setString(i, parameter.getKey());    }        @Override    public SexEnum getNullableResult(ResultSet rs, String columnName) throws SQLException {        System.out.println("columnName执行我");        // 根据数据库存储类型决定获取类型,本例子中数据库中存放String类型        String key = rs.getString(columnName);        if (rs.wasNull()) {            return null;        } else {            // 根据数据库中的key值,定位SexEnum子类            return SexEnum.getSexEnumByKey(key);        }    }        @Override    public SexEnum getNullableResult(ResultSet rs, int columnIndex) throws SQLException {        System.out.println("columnIndex执行我");        // 根据数据库存储类型决定获取类型,本例子中数据库中存放String类型        String key = rs.getString(columnIndex);        if (rs.wasNull()) {            return null;        } else {            // 根据数据库中的key值,定位SexEnum子类            return SexEnum.getSexEnumByKey(key);        }    }        @Override    public SexEnum getNullableResult(CallableStatement cs, int columnIndex) throws SQLException {        // 根据数据库存储类型决定获取类型,本例子中数据库中存放String类型        String key = cs.getString(columnIndex);        if (cs.wasNull()) {            return null;        } else {            // 根据数据库中的key值,定位SexEnum子类            return SexEnum.getSexEnumByKey(key);        }    }}

4、配置枚举处理器

mybatis配置

<?xml version="1.0" encoding="UTF-8"?>  <!DOCTYPE configuration PUBLIC "-//mybatis.org//DTD Config 3.0//EN"  "Http://mybatis.org/dtd/mybatis-3-config.dtd"><configuration>    <settings>        <!-- 打印sql语句 -->        <setting name="logImpl" value="STDOUT_LOGGING" />    </settings>    <typeHandlers>        <typeHandler handler="com.ahut.handler.SexEnumHandler"            javaType="com.ahut.core.enums.SexEnum" jdbcType="CHAR" />    </typeHandlers></configuration>

5、dao层

package com.ahut.mapper;import java.util.List;import java.util.Map;import com.ahut.entity.Demo;public interface DemoMapper {        void saveDemo(Map<String, Object> map) throws Exception;        List<Map<String, Object>> selectDemoList() throws Exception;        List<Demo> selectDemoList1() throws Exception;}

6、mapper文件

<?xml version="1.0" encoding="UTF-8"?><!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd"><mapper namespace="com.ahut.mapper.DemoMapper">    <!-- 保存 -->    <insert id="saveDemo" parameterType="map">        INSERT INTO DEMO        VALUES(replace(UUID(),'-',''),#{USER_NAME},#{AGE},#{BIRTH_DAY},#{SEX})    </insert>    <!-- 查询 -->    <select id="selectDemoList" resultType="map">        SELECT        ID,        USER_NAME,        AGE,        BIRTH_DAY,        SEX        FROM DEMO    </select>    <!-- 查询 -->    <select id="selectDemoList1" resultType="com.ahut.entity.Demo">        SELECT        ID,        USER_NAME USERNAME,        AGE,        BIRTH_DAY BIRTHDAY,        SEX        FROM DEMO    </select></mapper>

7、测试

package com.ahut.service;import java.util.Date;import java.util.HashMap;import java.util.List;import java.util.Map;import org.junit.Test;import org.junit.runner.RunWith;import org.springframework.beans.factory.annotation.Autowired;import org.springframework.boot.test.context.SpringBootTest;import org.springframework.test.context.junit4.SpringRunner;import com.ahut.core.enums.SexEnum;import com.ahut.entity.Demo;@SpringBootTest@RunWith(SpringRunner.class)public class DemoServiceTest {    @Autowired    private DemoService demoService;        @Test    public void testSelectDemoList1() throws Exception {        List<Demo> demoList = demoService.selectDemoList1();        for (Demo demo : demoList) {            System.out.println(demo);        }    }        @Test    public void testSelectDemoList() throws Exception {        List<Map<String, Object>> demoList = demoService.selectDemoList();        for (Map<String, Object> map : demoList) {            for (String key : map.keySet()) {                if (key.equals("BIRTH_DAY")) {                    Date birthDay = (Date) map.get(key);                    System.out.println(key + ":" + birthDay);                } else if (key.equals("AGE")) {                    int age = (int) map.get(key);                    System.out.println(key + ":" + age);                } else if (key.equals("SEX")) {                    SexEnum sex = (SexEnum) map.get(key);                    System.out.println(key + ":" + sex);                } else {                    String value = (String) map.get(key);                    System.out.println(key + ":" + value);                }            }        }    }        @Test    public void testSaveDemo() throws Exception {        Map<String, Object> map = new HashMap<>();        map.put("USER_NAME", "rick11");        map.put("AGE", 22);        map.put("BIRTH_DAY", new Date());        map.put("SEX", SexEnum.WOMAN);        demoService.saveDemo(map);    }}

执行testSaveDemo方法:

SexEnum.WOMAN被转换成了2存入到数据库中

Mybatis查询语句条件为枚举类型时报错怎么办

执行testSelectDemoList1方法:

数据库中的1、2成功被转换成了枚举

当resultType为包含枚举的实体类时,mybatis调用了枚举处理器

Mybatis查询语句条件为枚举类型时报错怎么办

Mybatis查询语句条件为枚举类型时报错怎么办

执行testSelectDemoList方法:

报错

由下图可知,resultType为map时,并没有调用枚举处理器

Mybatis查询语句条件为枚举类型时报错怎么办

Mybatis查询语句条件为枚举类型时报错怎么办

以上是“Mybatis查询语句条件为枚举类型时报错怎么办”这篇文章的所有内容,感谢各位的阅读!希望分享的内容对大家有帮助,更多相关知识,欢迎关注编程网精选频道!

--结束END--

本文标题: Mybatis查询语句条件为枚举类型时报错怎么办

本文链接: https://www.lsjlt.com/news/308952.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • Mybatis查询语句条件为枚举类型时报错怎么办
    这篇文章主要介绍Mybatis查询语句条件为枚举类型时报错怎么办,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!Mybatis查询语句条件为枚举类型报错通常我们对于数据库中一些枚举字段使用tinyInt类型,而java...
    99+
    2023-06-26
  • Mybatis查询语句条件为枚举类型时报错的解决
    目录Mybatis查询语句条件为枚举类型报错通常这个错误是Mybatis处理枚举类型1、枚举2、包含枚举的实体类3、书写枚举处理器4、配置枚举处理器5、dao层6、mapper文件7...
    99+
    2024-04-02
  • MyBatis查询无记录时返回值报错怎么办
    本篇内容主要讲解“MyBatis查询无记录时返回值报错怎么办”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“MyBatis查询无记录时返回值报错怎么办”吧!MyBatis查询无记录的返回值在MyB...
    99+
    2023-06-28
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作