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

MyBatis自定義TypeHandler如何解決字段映射問題

 更新時間:2023年12月06日 09:01:38   作者:串一串cc  
這篇文章主要介紹了MyBatis自定義TypeHandler如何解決字段映射問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教

MyBatis自定義TypeHandler字段映射

小林子:串哥

串一串:干哈啊,又來

小林子:如果MySQL一張表中一個字段存儲的數(shù)據(jù)格式是"1,2,3,4,5",也就是逗號分隔的,我如何能讓別的使用者在無感知的情況下,只用List<Integer>來傳輸和接收?持久層用的MyBatis。你滴明白我的意思嗎?

串一串:不明白

小林子:…

串一串:你知道MyBatis中有一個類叫BaseTypeHandler嗎?這個類可以滿足你的需求。

小林子:具體要怎么做?我有點懵,沒接觸過這個類,它是干嘛的?

串一串:我們來看個例子

創(chuàng)建一張表待用

create table qfant_message.demo (
	id int auto_increment primary key,
	name varchar(10) null,
	hobbies varchar(100) null
);

然后新建一個SpringBoot工程,在工程中引入mybatis-generator,我們使用它來生成Mapper文件,如果不會的話,自行谷歌,這里不做詳細講解,下一篇再說。

在生成Mapper文件之前,我們先定義一個處理字段hobbiesTypeHandler,命名為ListTypeHandler,這里問個問題:為什么不叫HobbiesTypeHandler呢?這樣應該和字段更加貼合啊。

原因是這個Handler不僅僅是能處理hobbies,它可以處理所有相同情況的任何表的任何字段。

這個類繼承自org.apache.ibatis.type.BaseTypeHandler

看下簡化后的內(nèi)容:

public abstract class BaseTypeHandler<T> extends TypeReference<T> implements TypeHandler<T> {

  @Override
  public void setParameter(PreparedStatement ps, int i, T parameter, JdbcType jdbcType) throws SQLException {
    if (parameter == null) {
      ps.setNull(i, jdbcType.TYPE_CODE);
    } else {
      setNonNullParameter(ps, i, parameter, jdbcType);
    }
  }

  @Override
  public T getResult(ResultSet rs, String columnName) throws SQLException {
    return getNullableResult(rs, columnName);
  }

  @Override
  public T getResult(ResultSet rs, int columnIndex) throws SQLException {
    return getNullableResult(rs, columnIndex);
  }

  @Override
  public T getResult(CallableStatement cs, int columnIndex) throws SQLException {
    return getNullableResult(cs, columnIndex);
  }

  public abstract void setNonNullParameter(PreparedStatement ps, int i, T parameter, JdbcType jdbcType) throws SQLException;

  public abstract T getNullableResult(ResultSet rs, String columnName) throws SQLException;

  public abstract T getNullableResult(ResultSet rs, int columnIndex) throws SQLException;

  public abstract T getNullableResult(CallableStatement cs, int columnIndex) throws SQLException;

}

我們可以看到,不論是通過哪一個getResult方法獲取數(shù)據(jù),都是去調(diào)用下面的幾個抽象方法,MyBatis幫我們實現(xiàn)了很多常用的類型的Handler,都在org.apache.ibatis.type包里面,截圖看下吧,免得以為在忽悠你

小林子:那這里面有沒有能滿足我這個需求的Handler?如果有的話我就直接用了

串一串:你去看看,這里我說一下怎么重復造輪子

根據(jù)上述內(nèi)容,我們就可以來寫ListTypeHandler了,在寫之前先整理一下思路:

因為我們實體類中hobbies屬性是java.util.List類型的,而數(shù)據(jù)庫表中hobbies字段是varchar類型的,所以我們需要在更新(插入)之前和查詢之后對數(shù)據(jù)進行一次轉(zhuǎn)換

  • 插入之前:將List中的數(shù)據(jù)轉(zhuǎn)換為以逗號分隔的字符串
  • 查詢之后:將逗號分隔的字符串轉(zhuǎn)換為List結(jié)構(gòu)

思路理順了

我們來看看具體的代碼

package cc.kevinlu.handler;

import java.sql.CallableStatement;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.stream.Collectors;

import org.apache.commons.lang3.StringUtils;
import org.apache.ibatis.type.BaseTypeHandler;
import org.apache.ibatis.type.JdbcType;
import org.apache.ibatis.type.MappedJdbcTypes;
import org.apache.ibatis.type.MappedTypes;

import com.qfant.sms.data.model.DemoDO;

@MappedJdbcTypes(value = { JdbcType.VARCHAR })
//① @MappedTypes(value = DemoDO.class)
public class ListTypeHandler extends BaseTypeHandler<List<Integer>> {
    @Override
    public void setNonNullParameter(PreparedStatement ps, int i, List<Integer> parameter, JdbcType jdbcType)
            throws SQLException {
        String d = parameter.stream().map(v -> String.valueOf(v)).collect(Collectors.joining(","));
        ps.setString(i, d);
    }

    @Override
    public List<Integer> getNullableResult(ResultSet rs, String columnName) throws SQLException {
        String values = rs.getString(columnName);
        return getResults(values);
    }

    @Override
    public List<Integer> getNullableResult(ResultSet rs, int columnIndex) throws SQLException {
        String values = rs.getString(columnIndex);
        return getResults(values);
    }

    @Override
    public List<Integer> getNullableResult(CallableStatement cs, int columnIndex) throws SQLException {
        String values = cs.getString(columnIndex);
        return getResults(values);
    }

    private List<Integer> getResults(String values) {
        if (StringUtils.isNotBlank(values)) {
            String[] data = values.split(",");
            return Arrays.stream(data).mapToInt(v -> Integer.parseInt(v)).boxed().collect(Collectors.toList());
        }
        return new ArrayList<>();
    }
}

然后生成對應的Mapper和DO實體類,剛才說了我們使用的是mybatis-generator,這里直接貼上<table>的相關(guān)配置

<table tableName="demo" domainObjectName="DemoDO" mapperName="DemoMapper"
       enableCountByExample="true"
       enableDeleteByExample="true" enableInsert="true" enableSelectByExample="true"
       enableUpdateByExample="true"
       selectByExampleQueryId="true" enableSelectByPrimaryKey="true">
  <generatedKey column="id" sqlStatement="MySql" identity="true"/>
  <columnOverride column="hobbies" property="hobbies" jdbcType="VARCHAR" javaType="java.util.List"
                  typeHandler="cc.kevinlu.handler.ListTypeHandler"/>
</table>

注意這里我們使用標簽<columnOverride>重寫了column的定義,這里一定要指明javaTypetypeHandlerjavaType的目的是讓生成的DemoDO的屬性hobbies聲明為java.util.List,如果不加該字段的話,默認會根據(jù)jdbcType="VARCHAR"生成java.lang.String類型,然后typeHandler指向我們剛創(chuàng)建的ListTypeHandler,這樣在生成DemoMapper.xml的時候,會在對應的字段上加上typeHandler,否則需要我們挨個兒位置的去修改,xml中的內(nèi)容如下:

<insert id="insert" parameterType="cc.kevinlu.data.model.DemoDO">
  <selectKey keyProperty="id" order="AFTER" resultType="java.lang.Integer">
    SELECT LAST_INSERT_ID()
  </selectKey>
  insert into demo (name, hobbies)
  values (
  		#{name,jdbcType=VARCHAR},
  		#{hobbies,jdbcType=VARCHAR,typeHandler=com.qfant.sms.handler.ListTypeHandler}
  )
</insert>

小林子:是不是這樣就可以直接使用了?

串一串:你有沒有注意到ListTypeHandler上有一個被注釋掉的注解,把那個注釋打開,然后value指向DO實體類即可,這個注釋的意思是指定該Handler映射的java類,value是一個數(shù)組,可以指定一組映射類,當然也可以不指定。即使指定了,也可以用于其他類型,然后@MappedJdbcTypes映射的是jdbc的類型

小林子:那現(xiàn)在是不是可以測試啦?走一波~

@Resource
private DemoMapper demoMapper;

@Test
public void index() {
  List<DemoDO> data = demoMapper.selectByExample(new DemoDOExample());
  data.forEach(System.out::println);
}

輸出:

Demo1DO [Hash = 3112387, id=1, name=123, hobbies=[1, 2, 3], serialVersionUID=1]
Demo1DO [Hash = 3294608, id=2, name=456, hobbies=[4, 5, 6], serialVersionUID=1]

總結(jié)一下

1.MyBatis之所以能解決MySQL字段和Java屬性之間的匹配,全都依賴于org.apache.ibatis.type.BaseTypeHandler<T>抽象類,在該類中定義了3個獲取結(jié)果的方法、1個更新的方法和4個抽象方法,我們可以自定義該抽象類來實現(xiàn)這個4個抽象方法進行Java類的屬性和表字段的映射,可以做一些相關(guān)的處理。

2.MyBatis在org.apache.ibatis.type包中定義了常用的字段映射Handler,并且在服務啟動的時候會在TypeHandlerRegistry構(gòu)造方法中將其注冊到一個Map中,而TypeHandlerRegistry是在MyBatis的核心類Configuration中進行的實例化

3.自定義的Handler可以全局通用,不受限于某一個字段或某一個Java類

4.在生成Mapper時使用<columnOverride>重寫column聲明,然后需要指定jdbcTypetypeHandler

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

相關(guān)文章

  • SpringBoot開發(fā)存儲服務器實現(xiàn)過程詳解

    SpringBoot開發(fā)存儲服務器實現(xiàn)過程詳解

    這篇文章主要為大家介紹了SpringBoot開發(fā)存儲服務器實現(xiàn)過程詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-12-12
  • Java 實戰(zhàn)項目錘煉之醫(yī)院門診收費管理系統(tǒng)的實現(xiàn)流程

    Java 實戰(zhàn)項目錘煉之醫(yī)院門診收費管理系統(tǒng)的實現(xiàn)流程

    讀萬卷書不如行萬里路,只學書上的理論是遠遠不夠的,只有在實戰(zhàn)中才能獲得能力的提升,本篇文章手把手帶你用java+html+jdbc+mysql實現(xiàn)一個醫(yī)院門診收費管理系統(tǒng),大家可以在過程中查缺補漏,提升水平
    2021-11-11
  • java實現(xiàn)文本文件刪除空行的示例分享

    java實現(xiàn)文本文件刪除空行的示例分享

    這篇文章主要介紹了java實現(xiàn)文本文件刪除空行的示例,需要的朋友可以參考下
    2014-04-04
  • 關(guān)于Java的ArrayList數(shù)組自動擴容機制

    關(guān)于Java的ArrayList數(shù)組自動擴容機制

    這篇文章主要介紹了關(guān)于Java的ArrayList數(shù)組自動擴容機制,ArrayList底層是基于數(shù)組實現(xiàn)的,是一個動態(tài)數(shù)組,自動擴容,不是線程安全的,只能用在單線程環(huán)境下,需要的朋友可以參考下
    2023-05-05
  • Spring中的AOP原理與使用詳解

    Spring中的AOP原理與使用詳解

    這篇文章主要介紹了Spring中的AOP原理與使用詳解,AOP意為面向切面編程,可以通過預編譯方式或運行期動態(tài)代理實現(xiàn)在不修改源代碼的情況下給程序動態(tài)統(tǒng)一添加功能的一種技術(shù),需要的朋友可以參考下
    2023-12-12
  • 詳解基于SpringBoot使用AOP技術(shù)實現(xiàn)操作日志管理

    詳解基于SpringBoot使用AOP技術(shù)實現(xiàn)操作日志管理

    這篇文章主要介紹了詳解基于SpringBoot使用AOP技術(shù)實現(xiàn)操作日志管理,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2019-11-11
  • Mybatis中typeAliases的使用

    Mybatis中typeAliases的使用

    這篇文章主要介紹了Mybatis中typeAliases的使用,需要的朋友可以參考下
    2017-08-08
  • ActiveMQ結(jié)合Spring收發(fā)消息的示例代碼

    ActiveMQ結(jié)合Spring收發(fā)消息的示例代碼

    這篇文章主要介紹了ActiveMQ結(jié)合Spring收發(fā)消息的示例代碼,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2018-10-10
  • SpringBoot文件上傳功能的實現(xiàn)方法

    SpringBoot文件上傳功能的實現(xiàn)方法

    這篇文章主要介紹了SpringBoot文件上傳功能的實現(xiàn),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2022-08-08
  • request如何獲取body的json數(shù)據(jù)

    request如何獲取body的json數(shù)據(jù)

    這篇文章主要介紹了request如何獲取body的json數(shù)據(jù)操作,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-06-06

最新評論