SpringBoot攻略十一、自定義ISqlInjector敛惊,添加通用方法updateAllColumnById

參考【SpringBoot攻略七渊鞋、集成mybatisplus實戰(zhàn)】,做如下修改:

1瞧挤、自定義sql注入器GeneralMybatisPlusSqlInjector

package com.javasgj.springboot.mybatisplus.config;

import java.util.List;

import com.baomidou.mybatisplus.core.injector.AbstractMethod;
import com.baomidou.mybatisplus.core.injector.DefaultSqlInjector;

/**
 * 自定義sql注入器锡宋,增加通用方法
 */
public class GeneralMybatisPlusSqlInjector extends DefaultSqlInjector {

    @Override
    public List<AbstractMethod> getMethodList() {
        
        List<AbstractMethod> methodList = super.getMethodList();
        
        // 根據(jù)id更新所有數(shù)據(jù)
        methodList.add(new UpdateAllColumnById());
        return methodList;
    }
}

2、方法對應(yīng)的實現(xiàn)類UpdateAllColumnById

package com.javasgj.springboot.mybatisplus.config;

import static java.util.stream.Collectors.joining;

import org.apache.ibatis.mapping.MappedStatement;
import org.apache.ibatis.mapping.SqlSource;

import com.baomidou.mybatisplus.annotation.FieldFill;
import com.baomidou.mybatisplus.core.injector.AbstractMethod;
import com.baomidou.mybatisplus.core.metadata.TableFieldInfo;
import com.baomidou.mybatisplus.core.metadata.TableInfo;
import com.baomidou.mybatisplus.core.toolkit.Constants;
import com.baomidou.mybatisplus.core.toolkit.StringPool;
import com.baomidou.mybatisplus.core.toolkit.StringUtils;
import com.baomidou.mybatisplus.core.toolkit.sql.SqlScriptUtils;

/**
 * 根據(jù)id更新所有數(shù)據(jù)
 */
public class UpdateAllColumnById extends AbstractMethod {

    @Override
    public MappedStatement injectMappedStatement(Class<?> mapperClass, Class<?> modelClass, TableInfo tableInfo) {
        
        GeneralMybatisPlusSqlMethod sqlMethod = GeneralMybatisPlusSqlMethod.UPDATE_ALL_COLUMN_BY_ID;
        String sql = String.format(sqlMethod.getSql(), tableInfo.getTableName(),
            sqlSet(false, false, tableInfo, Constants.ENTITY_SPOT),
            tableInfo.getKeyColumn(), Constants.ENTITY_SPOT + tableInfo.getKeyProperty(),
            new StringBuilder("<if test=\"et instanceof java.util.Map\">")
                .append("<if test=\"et.MP_OPTLOCK_VERSION_ORIGINAL!=null\">")
                .append(" AND ${et.MP_OPTLOCK_VERSION_COLUMN}=#{et.MP_OPTLOCK_VERSION_ORIGINAL}")
                .append("</if></if>"));
        
        SqlSource sqlSource = languageDriver.createSqlSource(configuration, sql, modelClass);
        return addUpdateMappedStatement(mapperClass, modelClass, sqlMethod.getMethod(), sqlSource);
    }

    @Override
    protected String sqlSet(boolean logic, boolean ew, TableInfo table, String prefix) {
        
        String newPrefix = prefix == null ? StringPool.EMPTY : prefix;
        String sqlScript = table.getFieldList().stream()
            .filter(i -> {
                return true;
            })
            .map(i -> {
                return this.getSqlSet(i, newPrefix);
            }).collect(joining(StringPool.NEWLINE));
            
        if (ew) {
            sqlScript += StringPool.NEWLINE;
            sqlScript += SqlScriptUtils.convertIf(SqlScriptUtils.unSafeParam(Constants.U_WRAPPER_SQL_SET),
                String.format("%s != null and %s != null", Constants.WRAPPER, Constants.U_WRAPPER_SQL_SET), false);
        }
        sqlScript = SqlScriptUtils.convertTrim(sqlScript, "SET", null, null, ",");
        return sqlScript;
    }
    
    public String getSqlSet(TableFieldInfo i, String prefix) {
        
        String newPrefix = prefix == null ? StringPool.EMPTY : prefix;
        String column = i.getColumn();
        String update = i.getUpdate();
        FieldFill fieldFill = i.getFieldFill();
        String el = i.getEl();
        
        // 默認(rèn): column=
        String sqlSet = column + StringPool.EQUALS;
        if (StringUtils.isNotEmpty(update)) {
            sqlSet += String.format(update, column);
        } else {
            sqlSet += SqlScriptUtils.safeParam(newPrefix + el);
        }
        sqlSet += StringPool.COMMA;
        if (fieldFill == FieldFill.UPDATE || fieldFill == FieldFill.INSERT_UPDATE) {
            // 不進(jìn)行 if 包裹
            return sqlSet;
        }
        return sqlSet;
    }
}

參考其他基本方法的實現(xiàn)類源碼如:UpdateById等等

3特恬、MybatisPlus自定義SQL方法枚舉類GeneralMybatisPlusSqlMethod

package com.javasgj.springboot.mybatisplus.config;

/**
 * MybatisPlus自定義SQL方法
 */
public enum GeneralMybatisPlusSqlMethod {
    
    /**
     * 修改
     */
    UPDATE_ALL_COLUMN_BY_ID("updateAllColumnById", "根據(jù)ID更新所有數(shù)據(jù)", "<script>\nUPDATE %s %s WHERE %s=#{%s} %s\n</script>");

    private final String method;
    private final String desc;
    private final String sql;

    GeneralMybatisPlusSqlMethod(String method, String desc, String sql) {
        this.method = method;
        this.desc = desc;
        this.sql = sql;
    }

    public String getMethod() {
        return method;
    }

    public String getDesc() {
        return desc;
    }

    public String getSql() {
        return sql;
    }
}

4执俩、MybatisPlus配置類,加載自定義sql注入器

package com.javasgj.springboot.mybatisplus.config;

import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

import com.baomidou.mybatisplus.core.injector.ISqlInjector;
import com.baomidou.mybatisplus.extension.plugins.PaginationInterceptor;

/**
 * MybatisPlus配置類
 */
@Configuration
public class MybatisPlusConfig {

    /**
     * 分頁插件
     * 或者在mybatis-config.xml配置:
     *  <plugins>  
     *      <!-- mybatisplus分頁攔截器 -->
     *      <plugin interceptor="com.baomidou.mybatisplus.extension.plugins.PaginationInterceptor">
     *      </plugin>  
     *  </plugins> 
     */
    @Bean
    public PaginationInterceptor paginationInterceptor() {
        
        return new PaginationInterceptor();
    }

    /**
     * 自定義sql注入器
     * 或者application.properties配置:
     * mybatis-plus.globalConfig.sqlInjector=com.javasgj.springboot.mybatisplus.config.GeneralMybatisPlusSqlInjector
     */
    @Bean
    public ISqlInjector iSqlInjector() {
        
        return new GeneralMybatisPlusSqlInjector();
    }

    /**
     * sql性能分析插件鸵鸥,輸出sql語句及所需時間
     */
    /*@Bean
    @Profile({"dev","test"})// 設(shè)置 dev test 環(huán)境開啟
    public PerformanceInterceptor performanceInterceptor() {
        return new PerformanceInterceptor();
    }*/
    
    /**
     * 樂觀鎖插件
     */
    /*@Bean
    public OptimisticLockerInterceptor optimisticLockerInterceptor(){
        return new OptimisticLockerInterceptor();
    }*/
}

5奠滑、自定義基礎(chǔ)Mapper繼承BaseMapper

package com.javasgj.springboot.mybatisplus.dao;

import org.apache.ibatis.annotations.Param;

import com.baomidou.mybatisplus.core.mapper.BaseMapper;
import com.baomidou.mybatisplus.core.toolkit.Constants;

public interface GeneralBaseMapper<T> extends BaseMapper<T> {

    /**
     * 根據(jù)id更新所有數(shù)據(jù)
     * @param entity
     * @return
     */
    int updateAllColumnById(@Param(Constants.ENTITY) T entity);
}

6、自定義基礎(chǔ)service繼承IService及實現(xiàn)類

package com.javasgj.springboot.mybatisplus.service;

import java.util.Collection;

import com.baomidou.mybatisplus.extension.service.IService;

public interface GeneralService<T> extends IService<T> {

    /**
     * 根據(jù)ID更新所有數(shù)據(jù)
     * @param entity
     * @return
     */
    boolean updateAllColumnById(T entity);
    
    /**
     * 根據(jù)ID批量更新所有數(shù)據(jù)
     * @param entityList
     * @return
     */
    default boolean updateAllColumnBatchById(Collection<T> entityList) {
        return updateAllColumnBatchById(entityList, 30);
    }
    
    /**
     * 根據(jù)ID批量更新所有數(shù)據(jù)
     * @param entityList
     * @param batchSize
     * @return
     */
    boolean updateAllColumnBatchById(Collection<T> entityList, int batchSize);
}

實現(xiàn)類

package com.javasgj.springboot.mybatisplus.service;

import java.util.Collection;

import org.apache.ibatis.binding.MapperMethod;
import org.apache.ibatis.session.SqlSession;
import org.springframework.transaction.annotation.Transactional;

import com.baomidou.mybatisplus.core.toolkit.CollectionUtils;
import com.baomidou.mybatisplus.core.toolkit.Constants;
import com.baomidou.mybatisplus.core.toolkit.sql.SqlHelper;
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
import com.javasgj.springboot.mybatisplus.config.GeneralMybatisPlusSqlMethod;
import com.javasgj.springboot.mybatisplus.dao.GeneralBaseMapper;

public class GeneralServiceImpl<M extends GeneralBaseMapper<T>, T> extends ServiceImpl<GeneralBaseMapper<T>, T> implements GeneralService<T> {

    /**
     * 獲取SqlStatement
     *
     * @param sqlMethod
     * @return
     */
    protected String sqlStatement(GeneralMybatisPlusSqlMethod generalMybatisPlusSqlMethod) {
        return SqlHelper.table(currentModelClass()).getSqlStatement(generalMybatisPlusSqlMethod.getMethod());
    }
    
    @Transactional(rollbackFor = Exception.class)
    @Override
    public boolean updateAllColumnById(T entity) {
        return retBool(baseMapper.updateAllColumnById(entity));
    }

    @Transactional(rollbackFor = Exception.class)
    @Override
    public boolean updateAllColumnBatchById(Collection<T> entityList, int batchSize) {
        if (CollectionUtils.isEmpty(entityList)) {
            throw new IllegalArgumentException("Error: entityList must not be empty");
        }
        int i = 0;
        String sqlStatement = sqlStatement(GeneralMybatisPlusSqlMethod.UPDATE_ALL_COLUMN_BY_ID);
        try (SqlSession batchSqlSession = sqlSessionBatch()) {
            for (T anEntityList : entityList) {
                MapperMethod.ParamMap<T> param = new MapperMethod.ParamMap<>();
                param.put(Constants.ENTITY, anEntityList);
                batchSqlSession.update(sqlStatement, param);
                if (i >= 1 && i % batchSize == 0) {
                    batchSqlSession.flushStatements();
                }
                i++;
            }
            batchSqlSession.flushStatements();
        }
        return true;
    }
}

然后所有的mapper和servcie繼承我們自定義擴(kuò)展的基礎(chǔ)mapper和service
好了妒穴,大家開始測試測試吧宋税,參考【原 SpringBoot秘籍七、集成mybatisplus實戰(zhàn)】

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末讼油,一起剝皮案震驚了整個濱河市杰赛,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌矮台,老刑警劉巖乏屯,帶你破解...
    沈念sama閱讀 212,686評論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異瘦赫,居然都是意外死亡辰晕,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,668評論 3 385
  • 文/潘曉璐 我一進(jìn)店門确虱,熙熙樓的掌柜王于貴愁眉苦臉地迎上來含友,“玉大人,你說我怎么就攤上這事校辩【轿剩” “怎么了?”我有些...
    開封第一講書人閱讀 158,160評論 0 348
  • 文/不壞的土叔 我叫張陵宜咒,是天一觀的道長惠赫。 經(jīng)常有香客問我,道長故黑,這世上最難降的妖魔是什么儿咱? 我笑而不...
    開封第一講書人閱讀 56,736評論 1 284
  • 正文 為了忘掉前任庭砍,我火速辦了婚禮,結(jié)果婚禮上概疆,老公的妹妹穿的比我還像新娘逗威。我一直安慰自己,他們只是感情好岔冀,可當(dāng)我...
    茶點故事閱讀 65,847評論 6 386
  • 文/花漫 我一把揭開白布凯旭。 她就那樣靜靜地躺著,像睡著了一般使套。 火紅的嫁衣襯著肌膚如雪罐呼。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 50,043評論 1 291
  • 那天侦高,我揣著相機(jī)與錄音嫉柴,去河邊找鬼。 笑死奉呛,一個胖子當(dāng)著我的面吹牛计螺,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播瞧壮,決...
    沈念sama閱讀 39,129評論 3 410
  • 文/蒼蘭香墨 我猛地睜開眼登馒,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了咆槽?” 一聲冷哼從身側(cè)響起陈轿,我...
    開封第一講書人閱讀 37,872評論 0 268
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎秦忿,沒想到半個月后麦射,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,318評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡灯谣,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,645評論 2 327
  • 正文 我和宋清朗相戀三年潜秋,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片胎许。...
    茶點故事閱讀 38,777評論 1 341
  • 序言:一個原本活蹦亂跳的男人離奇死亡峻呛,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出呐萨,到底是詐尸還是另有隱情杀饵,我是刑警寧澤莽囤,帶...
    沈念sama閱讀 34,470評論 4 333
  • 正文 年R本政府宣布谬擦,位于F島的核電站,受9級特大地震影響朽缎,放射性物質(zhì)發(fā)生泄漏惨远。R本人自食惡果不足惜谜悟,卻給世界環(huán)境...
    茶點故事閱讀 40,126評論 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望北秽。 院中可真熱鬧葡幸,春花似錦、人聲如沸贺氓。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,861評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽辙培。三九已至蔑水,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間扬蕊,已是汗流浹背搀别。 一陣腳步聲響...
    開封第一講書人閱讀 32,095評論 1 267
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留尾抑,地道東北人歇父。 一個月前我還...
    沈念sama閱讀 46,589評論 2 362
  • 正文 我出身青樓,卻偏偏與公主長得像再愈,于是被迫代替她去往敵國和親榜苫。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 43,687評論 2 351