Mybatis 查詢語句條件為枚舉類型時報錯的解決
Mybatis查詢語句條件為枚舉類型報錯
通常我們對于數(shù)據(jù)庫中一些枚舉字段使用tinyInt類型,而java對象對應(yīng)的字段很多時候會為了方便定義成short或者int。但這樣顯然不美觀方便,讓后面維護(hù)的人摳破腦袋找你的常量定義在哪兒,要是沒有注釋簡直讓人崩潰。時間久后,沒有人知道這里面的值。只能一行行讀源碼。
優(yōu)雅的程序員當(dāng)然想到了優(yōu)雅的枚舉,而mybatis“強(qiáng)大”的枚舉類型處理器EnumOrdinalTypeHandler相信都不陌生。
然而配置枚舉處理器花了九牛二虎之力改好原來的mapper運行測試用例全在報錯。而插入、部分查詢卻沒報錯。這時進(jìn)程進(jìn)行到一半讓人崩潰想要放棄。
通常這個錯誤是
"failed to invoke constructor for handler class org.apache.ibatis.type.EnumOrdinalTypeHandler”
原因是因為該死的查詢條件使用枚舉對象作為條件,無論你用selectExample還是其他的select,當(dāng)條件where enum = #{enum}時就會報錯。不要懷疑自己是不是EnumOrdinalTypeHandler沒配對,如果沒配對那一定會是所有的查詢接口都會報錯。
stackoverflow上只有一條相關(guān)問題。為什么這么少?這不是很常見的錯誤嗎?jpa或hibernate就能很優(yōu)雅的使用枚舉啊。原因嘛,老外們很少用半自動的mybatis框架。只有國內(nèi)奉為圭臬,原因嘛當(dāng)然是聽說人家阿里就用mybatis,所以一定是好的。也不看自己的業(yè)務(wù)到底是否真正觸及到要提升sql性能的地步。
話說回來,目前給出來的答案似乎是mybatis的bug,但對于mybatis這種半自動框架這不一定是bug。
解決辦法很簡單粗暴,把where enum = #{enum}條件換成where enum in (***)萬事大吉。但熟悉的同學(xué)已經(jīng)發(fā)現(xiàn)了。這樣的性能顯然不如=。用short和int的同學(xué)肯定又開心了。看吧我就說數(shù)據(jù)庫什么類型就用什么類型,枚舉就是垃圾。說這話的同學(xué)顯然還不習(xí)慣封裝、規(guī)范這一套,更喜歡隨心所欲的感覺。
今天的教訓(xùn)就到這。
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<String, SexEnum> sexEnumMap = new HashMap<>();
static {
for (SexEnum sexEnum : SexEnum.values()) {
sexEnumMap.put(sexEnum.getKey(), sexEnum);
}
}
/**
* 私有化構(gòu)造函數(shù)
*
* @param key
* @param value
*/
private SexEnum(String key, String value) {
this.key = key;
this.value = value;
}
/**
*
* @Title: getSexEnumByKey
* @Description: 依據(jù)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 extends BaseTypeHandler<SexEnum> {
/**
* 用于定義設(shè)置參數(shù)時,該如何把Java類型的參數(shù)轉(zhuǎn)換為對應(yīng)的數(shù)據(jù)庫類型
*/
@Override
public void setNonNullParameter(PreparedStatement ps, int i, SexEnum parameter, JdbcType jdbcType)
throws SQLException {
// baseTypeHandler已經(jīng)幫我們做了parameter的null判斷
// 第二個參數(shù) : 存入到數(shù)據(jù)庫中的值
ps.setString(i, parameter.getKey());
}
/**
* 用于定義通過字段名稱獲取字段數(shù)據(jù)時,如何把數(shù)據(jù)庫類型轉(zhuǎn)換為對應(yīng)的Java類型
*/
@Override
public SexEnum getNullableResult(ResultSet rs, String columnName) throws SQLException {
System.out.println("columnName執(zhí)行我");
// 根據(jù)數(shù)據(jù)庫存儲類型決定獲取類型,本例子中數(shù)據(jù)庫中存放String類型
String key = rs.getString(columnName);
if (rs.wasNull()) {
return null;
} else {
// 根據(jù)數(shù)據(jù)庫中的key值,定位SexEnum子類
return SexEnum.getSexEnumByKey(key);
}
}
/**
* 用于定義通過字段索引獲取字段數(shù)據(jù)時,如何把數(shù)據(jù)庫類型轉(zhuǎn)換為對應(yīng)的Java類型
*/
@Override
public SexEnum getNullableResult(ResultSet rs, int columnIndex) throws SQLException {
System.out.println("columnIndex執(zhí)行我");
// 根據(jù)數(shù)據(jù)庫存儲類型決定獲取類型,本例子中數(shù)據(jù)庫中存放String類型
String key = rs.getString(columnIndex);
if (rs.wasNull()) {
return null;
} else {
// 根據(jù)數(shù)據(jù)庫中的key值,定位SexEnum子類
return SexEnum.getSexEnumByKey(key);
}
}
/**
* 用定義調(diào)用存儲過程后,如何把數(shù)據(jù)庫類型轉(zhuǎn)換為對應(yīng)的Java類型
*/
@Override
public SexEnum getNullableResult(CallableStatement cs, int columnIndex) throws SQLException {
// 根據(jù)數(shù)據(jù)庫存儲類型決定獲取類型,本例子中數(shù)據(jù)庫中存放String類型
String key = cs.getString(columnIndex);
if (cs.wasNull()) {
return null;
} else {
// 根據(jù)數(shù)據(jù)庫中的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;
/**
*
* @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<String, Object> map) throws Exception;
/**
*
* @Title: selectDemoList
* @Description: 查詢
* @return
* @throws Exception
*/
List<Map<String, Object>> selectDemoList() throws Exception;
/**
*
* @Title: selectDemoList1
* @Description: 查詢
* @return
* @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;
/**
*
* @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<Demo> demoList = demoService.selectDemoList1();
for (Demo demo : demoList) {
System.out.println(demo);
}
}
/**
*
* @Title: testSelectDemoList
* @Description:
* @throws Exception
*/
@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);
}
}
}
}
/**
*
* @Title: testSaveDemo
* @Description:
* @throws Exception
*/
@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);
}
}執(zhí)行testSaveDemo方法:
SexEnum.WOMAN被轉(zhuǎn)換成了2存入到數(shù)據(jù)庫中

執(zhí)行testSelectDemoList1方法:
數(shù)據(jù)庫中的1、2成功被轉(zhuǎn)換成了枚舉
當(dāng)resultType為包含枚舉的實體類時,mybatis調(diào)用了枚舉處理器


執(zhí)行testSelectDemoList方法:
報錯
由下圖可知,resultType為map時,并沒有調(diào)用枚舉處理器


以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
解決RestTemplate 的getForEntity調(diào)用接口亂碼的問題
這篇文章主要介紹了解決RestTemplate 的getForEntity調(diào)用接口亂碼的問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-08-08
java集合中的迭代器Iterator和數(shù)組內(nèi)置方法及常見的報錯解決方案
文章介紹了Java集合框架中迭代器(Iterator)的使用,以及數(shù)組和集合的內(nèi)置方法,重點解釋了在遍歷集合時刪除元素時可能出現(xiàn)的`ConcurrentModificationException`異常,并說明了如何正確地使用迭代器來刪除集合中的元素,感興趣的朋友跟隨小編一起看看吧2025-02-02
SpringBoot深入刨析數(shù)據(jù)層技術(shù)
這篇文章主要介紹了SpringBoot數(shù)據(jù)層技術(shù)的解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-08-08
ElasticSearch如何設(shè)置某個字段不分詞淺析
最近在學(xué)習(xí)ElasticSearch官方文檔過程中發(fā)現(xiàn)的某個問題,記錄一下 希望能幫助到后面的朋友,下面這篇文章主要給大家介紹了關(guān)于ElasticSearch如何設(shè)置某個字段不分詞的相關(guān)資料,需要的朋友可以參考下2022-04-04
Spring Cloud Feign接口返回流的實現(xiàn)
這篇文章主要介紹了Spring Cloud Feign接口返回流的實現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-10-10

