多平台统一管理软件接口,如何实现多平台统一管理软件接口
401
2022-09-02
Mybatis 查询语句条件为枚举类型时报错的解决
目录Mybatis查询语句条件为枚举类型报错通常这个错误是Mybatis处理枚举类型1、枚举2、包含枚举的实体类3、书写枚举处理器4、配置枚举处理器5、dao层6、mapper文件7、测试
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;
/**
*
* @ClassName: SexEnum
* @Description: 性别枚举
* @author cheng
* @date 2017年11月20日 下午8:32:27
*/
public enum SexEnum {
MAN("1", "男"), WOMAN("2", "女");
private String key;
private String value;
private static Map
static {
for (SexEnum sexEnum : SexEnum.values()) {
sexEnumMap.put(sexEnum.getKey(), sexEnum);
}
}
/**
* 私有化构造函数
*
* @param key
* @param value
*/
private SexEnum(String key, String value) {
http:// this.key = key;
this.value = value;
}
/**
*
* @Title: getSexEnumByKey
* @Description: 依据key获取枚举
* @param key
* @return
*/
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;
/**
*
* @ClassName: Demo
* @Description:
* @author cheng
* @date 2017年11月21日 下午8:32:59
*/
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;
/**
*
* @ClassName: EnumHandler
* @Description:
* @author cheng
* @date 2017年11月20日 下午8:41:12
*/
public class SexEnumHandler exteniqNSLds BaseTypeHandler
/**
* 用于定义设置参数时,该如何把Java类型的参数转换为对应的数据库类型
*/
@Override
public void setNonNullParameter(PreparedStatement ps, int i, SexEnum parameter, JdbcType jdbcType)
throws SQLException {
// baseTypeHandler已经帮我们做了parameter的null判断
// 第二个参数 : 存入到数据库中的值
ps.setString(i, parameter.getKey());
}
/**
* 用于定义通过字段名称获取字段数据时,如何把数据库类型转换为对应的Java类型
*/
@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);
}
}
/**
* 用于定义通过字段索引获取字段数据时,如何把数据库类型转换为对应的Java类型
*/
@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);
}
}
/**
* 用定义调用存储过程后,如何把数据库类型转换为对应的Java类型
*/
@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配置
"http://mybatis.org/dtd/mybatis-3-config.dtd">
javaType="com.ahut.core.enums.SexEnum" jdbcType="CHAR" /> 5、dao层 package com.ahut.mapper; import java.util.List; import java.util.Map; import com.ahut.entity.Demo; /** * * @ClassName: DemoMapper * @Description: * @author cheng * @date 2017年11月16日 下午9:10:38 */ public interface DemoMapper { /** * * @Title: saveDemo * @Description: 保存 * @param map * @throws Exception */ void saveDemo(Map /** * * @Title: selectDemoList * @Description: 查询 * @return * @throws Exception */ List /** * * @Title: selectDemoList1 * @Description: 查询 * @return * @throws Exception */ List } 6、mapper文件
INSERT INTO DEMO VALUES(replace(UUID(),'-',''),#{USER_NAME},#{AGE},#{BIRTH_DAY},#{SEX}) SELECT ID, USER_NAME, AGE, BIRTH_DAY, SEX FROM DEMO SELECT ID, USER_NAME USERNAME, AGE, BIRTH_DAY BIRTHDAY, SEX FROM DEMO 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; /** * * @ClassName: DemoServiceTest * @Description: * @author cheng * @date 2017年11月16日 下午9:28:56 */ @SpringBootTest @RunWith(SpringRunner.class) public class DemoServiceTest { @Autowired private DemoService demoService; /** * * @Title: testSelectDemoList1 * @Description: * @throws Exception */ @Test public void testSelectDemoList1() throws Exception { List for (Demo demo : demoList) { System.out.println(demo); } } /** * * @Title: testSelectDemoList * @Description: * @throws Exception */ @Test public void testSelectDemoList() throws Exception { List for (Map 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); } } } } /** * * @Title: testSaveDemo * @Description: * @throws Exception */ @Test public void testSaveDemo() throws Exception { Map 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存入到数据库中 执行testSelectDemoList1方法: 数据库中的1、2成功被转换成了枚举 当resultType为包含枚举的实体类时,mybatis调用了枚举处理器 执行testSelectDemoList方法: 报错 由下图可知,resultType为map时,并没有调用枚举处理器
javaType="com.ahut.core.enums.SexEnum" jdbcType="CHAR" />
5、dao层
package com.ahut.mapper;
import java.util.List;
import java.util.Map;
import com.ahut.entity.Demo;
/**
*
* @ClassName: DemoMapper
* @Description:
* @author cheng
* @date 2017年11月16日 下午9:10:38
*/
public interface DemoMapper {
/**
*
* @Title: saveDemo
* @Description: 保存
* @param map
* @throws Exception
*/
void saveDemo(Map
/**
*
* @Title: selectDemoList
* @Description: 查询
* @return
* @throws Exception
*/
List
/**
*
* @Title: selectDemoList1
* @Description: 查询
* @return
* @throws Exception
*/
List
}
6、mapper文件
INSERT INTO DEMO
VALUES(replace(UUID(),'-',''),#{USER_NAME},#{AGE},#{BIRTH_DAY},#{SEX})
SELECT
ID,
USER_NAME,
AGE,
BIRTH_DAY,
SEX
FROM DEMO
SELECT
ID,
USER_NAME USERNAME,
AGE,
BIRTH_DAY BIRTHDAY,
SEX
FROM DEMO
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;
/**
*
* @ClassName: DemoServiceTest
* @Description:
* @author cheng
* @date 2017年11月16日 下午9:28:56
*/
@SpringBootTest
@RunWith(SpringRunner.class)
public class DemoServiceTest {
@Autowired
private DemoService demoService;
/**
*
* @Title: testSelectDemoList1
* @Description:
* @throws Exception
*/
@Test
public void testSelectDemoList1() throws Exception {
List
for (Demo demo : demoList) {
System.out.println(demo);
}
}
/**
*
* @Title: testSelectDemoList
* @Description:
* @throws Exception
*/
@Test
public void testSelectDemoList() throws Exception {
List
for (Map
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);
}
}
}
}
/**
*
* @Title: testSaveDemo
* @Description:
* @throws Exception
*/
@Test
public void testSaveDemo() throws Exception {
Map
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存入到数据库中
执行testSelectDemoList1方法:
数据库中的1、2成功被转换成了枚举
当resultType为包含枚举的实体类时,mybatis调用了枚举处理器
执行testSelectDemoList方法:
报错
由下图可知,resultType为map时,并没有调用枚举处理器
版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。
发表评论
暂时没有评论,来抢沙发吧~