欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Mybatis 查詢語句條件為枚舉類型時(shí)報(bào)錯(cuò)的解決

 更新時(shí)間:2022年01月19日 16:35:23   作者:顧人間譏詰  
這篇文章主要介紹了Mybatis 查詢語句條件為枚舉類型時(shí)報(bào)錯(cuò)的解決方案,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教

Mybatis查詢語句條件為枚舉類型報(bào)錯(cuò)

通常我們對于數(shù)據(jù)庫中一些枚舉字段使用tinyInt類型,而java對象對應(yīng)的字段很多時(shí)候會為了方便定義成short或者int。但這樣顯然不美觀方便,讓后面維護(hù)的人摳破腦袋找你的常量定義在哪兒,要是沒有注釋簡直讓人崩潰。時(shí)間久后,沒有人知道這里面的值。只能一行行讀源碼。

優(yōu)雅的程序員當(dāng)然想到了優(yōu)雅的枚舉,而mybatis“強(qiáng)大”的枚舉類型處理器EnumOrdinalTypeHandler相信都不陌生。

然而配置枚舉處理器花了九牛二虎之力改好原來的mapper運(yùn)行測試用例全在報(bào)錯(cuò)。而插入、部分查詢卻沒報(bào)錯(cuò)。這時(shí)進(jìn)程進(jìn)行到一半讓人崩潰想要放棄。

通常這個(gè)錯(cuò)誤是

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

原因是因?yàn)樵撍赖牟樵儣l件使用枚舉對象作為條件,無論你用selectExample還是其他的select,當(dāng)條件where enum = #{enum}時(shí)就會報(bào)錯(cuò)。不要懷疑自己是不是EnumOrdinalTypeHandler沒配對,如果沒配對那一定會是所有的查詢接口都會報(bào)錯(cuò)。

stackoverflow上只有一條相關(guān)問題。為什么這么少?這不是很常見的錯(cuò)誤嗎?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é)肯定又開心了??窗晌揖驼f數(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、包含枚舉的實(shí)體類

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ù)時(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判斷
        // 第二個(gè)參數(shù) : 存入到數(shù)據(jù)庫中的值
        ps.setString(i, parameter.getKey());
    }
    /**
     * 用于定義通過字段名稱獲取字段數(shù)據(jù)時(shí),如何把數(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í),如何把數(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為包含枚舉的實(shí)體類時(shí),mybatis調(diào)用了枚舉處理器

這里寫圖片描述

這里寫圖片描述

執(zhí)行testSelectDemoList方法:

報(bào)錯(cuò)

由下圖可知,resultType為map時(shí),并沒有調(diào)用枚舉處理器

這里寫圖片描述

這里寫圖片描述

以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。

相關(guān)文章

  • JVM虛擬機(jī)查找類文件的順序方法

    JVM虛擬機(jī)查找類文件的順序方法

    下面小編就為大家分享一篇JVM虛擬機(jī)查找類文件的順序方法,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2018-01-01
  • CentOS 7系統(tǒng)下配置自定義JDK的教程

    CentOS 7系統(tǒng)下配置自定義JDK的教程

    這篇文章主要給大家介紹了在CentOS 7系統(tǒng)下配置自定義JDK的教程,文中將配置的方法教程介紹的非常詳細(xì),對大家具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來一起看看吧。
    2017-06-06
  • Java 多線程死鎖的產(chǎn)生以及如何避免死鎖

    Java 多線程死鎖的產(chǎn)生以及如何避免死鎖

    這篇文章主要介紹了Java 多線程死鎖的產(chǎn)生以及如何避免死鎖,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-09-09
  • Java判空的一些常見方法

    Java判空的一些常見方法

    這篇文章主要給大家分享介紹了Java判空的一些常見方法,在程序中必須進(jìn)行嚴(yán)格的判空處理,避免對空對象的異常操作,文中通過實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2023-07-07
  • JavaWeb禁止瀏覽器緩存當(dāng)前Web頁面的方法

    JavaWeb禁止瀏覽器緩存當(dāng)前Web頁面的方法

    所謂瀏覽器緩存,是指當(dāng)?shù)谝淮卧L問網(wǎng)頁時(shí),瀏覽器會將這些網(wǎng)頁緩存到本地,當(dāng)下一次再訪問這些被緩存的網(wǎng)頁時(shí),瀏覽器就會直接從本地讀取這些網(wǎng)頁的內(nèi)容,而無需再從網(wǎng)絡(luò)上獲取
    2017-11-11
  • IntelliJ IDEA 的 Spring 項(xiàng)目如何查看 @Value 的配置和值(方法詳解)

    IntelliJ IDEA 的 Spring 項(xiàng)目如何查看 @Value 的配置和值(方法詳解)

    這篇文章主要介紹了IntelliJ IDEA 的 Spring 項(xiàng)目如何查看 @Value 的配置和值,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-10-10
  • Spring Boot+maven打war包的方法

    Spring Boot+maven打war包的方法

    這篇文章主要介紹了Spring Boot+maven打war包的方法,本文通過實(shí)例代碼相結(jié)合的形式給大家介紹的非常詳細(xì),需要的朋友參考下吧
    2018-05-05
  • IDEA中已配置阿里鏡像但maven無法下載jar包的問題及解決方法

    IDEA中已配置阿里鏡像但maven無法下載jar包的問題及解決方法

    這篇文章主要介紹了IDEA中已配置阿里鏡像但maven無法下載jar包的問題,本文給大家分享解決方法,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-08-08
  • Java實(shí)現(xiàn)用戶管理系統(tǒng)

    Java實(shí)現(xiàn)用戶管理系統(tǒng)

    這篇文章主要為大家詳細(xì)介紹了Java實(shí)現(xiàn)用戶管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-02-02
  • java去除中文括號小括號,或者英文括號的實(shí)例代碼

    java去除中文括號小括號,或者英文括號的實(shí)例代碼

    這篇文章主要介紹了java去除中文括號小括號,或者英文括號的實(shí)例代碼,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-09-09

最新評論