Spring Boot MyBatis Sql攔截器(自定義注解+反射)

業(yè)務(wù)場景

  • 公司APP需要將主模塊拆分成多個(gè)APP給代理商運(yùn)營
  • 不同代理商代理的APP產(chǎn)生的數(shù)據(jù)需根據(jù)對應(yīng)公司進(jìn)行區(qū)分
  • 公司總數(shù)據(jù)庫需同步管理所有代理商運(yùn)營的數(shù)據(jù)

設(shè)計(jì)思想

  • 設(shè)計(jì)在最小修改原則保證產(chǎn)品業(yè)務(wù)無大變動(dòng)只對表進(jìn)行新增COMPAN_ID字段進(jìn)行區(qū)分
  • 綜合以上需求場景產(chǎn)生的思路決定采用低耦合舵变、高復(fù)用進(jìn)行架構(gòu)設(shè)計(jì)
  • 通過MyBatis攔截器Sql進(jìn)行處理印颤,采用類自定義注解+反射的形式

代碼實(shí)現(xiàn)

創(chuàng)建自定義注解 HmwCompanyAnnotation

@Target({ElementType.METHOD,ElementType.PARAMETER})
@Retention(RetentionPolicy.RUNTIME)
public @interface HmwCompanyAnnotation {

    /**
     * 公司ID(暫未啟用)
     * @return
     */
    String CompanyId() default "";

    /**
     * 別名(針對復(fù)雜查詢備用處理屬性)
     * @return
     */
    String Alias() default "";

}

MyBatis的Mapper的實(shí)現(xiàn)示例
接口方法上添加 自定義注解@HmwCompanyAnnotation()

@Mapper
public interface SmsCodeDao  {

    /**
     * 新增
     *
     * @param smsCode
     * @return
     */
    @Options(useGeneratedKeys = true, keyProperty = "smsCodeId" , keyColumn = "SMS_CODE_ID")
    @Insert("insert into hmw_sms_code(SMS_TYPE, ... 省略字段..., COMPANY_ID) " +
            " values(#{smsType},...省略字段... ,#{companyId})")
    int save(SmsCode smsCode);


    /**
     * 查詢用戶每天發(fā)送的驗(yàn)證碼數(shù)量
     *
     * @param sendTel 手機(jī)號
     * @return
     */
    @HmwCompanyAnnotation()
    @Select("  select " +
            "     count(0) as count_num " +
            "  from hmw_sms_code" +
            "  where " +
            "       SEND_TEL = #{sendTel} " +
            "   and TO_DAYS(NOW()) = TO_DAYS(SEND_TIME)  ")
    Long findTelTodayCount(String sendTel );


    /**
     * 根據(jù)短信類型查詢是否已使用獲取已過期
     *
     * @param sendTel 手機(jī)號
     * @param sendCode 驗(yàn)證碼
     * @param smsType 短信類型
     * @param nowTime 當(dāng)前服務(wù)器時(shí)間(請用服務(wù)器New出來的時(shí)間 不要用NOW())
     * @return
     */
    @HmwCompanyAnnotation()
    @Select(" select   " +
            "  count(0) as count_num   " +
            " from   " +
            "  hmw_sms_code  " +
            " where   " +
            "      SEND_TEL = #{sendTel,jdbcType=VARCHAR}   " +
            "  and SEND_CODE = #{sendCode,jdbcType=VARCHAR}  " +
            "  and SMS_STATE = 'N'  " +
            "  and SMS_TYPE  = #{smsType,jdbcType=VARCHAR}  " +
            "  and SEND_TIME >= DATE_SUB(#{sendTime},INTERVAL 5  MINUTE) ")
    Long findSmsCodeByTel(String sendTel , String sendCode , String smsType , Date nowTime);



}

MyBatis 攔截器具體實(shí)現(xiàn)

import com.hmw.annotation.HmwCompanyAnnotation;
import com.hmw.base.Const;
import com.hmw.utils.common.ReflectHelper;
import org.apache.ibatis.executor.statement.RoutingStatementHandler;
import org.apache.ibatis.executor.statement.StatementHandler;
import org.apache.ibatis.mapping.BoundSql;
import org.apache.ibatis.mapping.MappedStatement;
import org.apache.ibatis.plugin.*;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Component;
import java.lang.reflect.Method;
import java.sql.Connection;
import java.util.Properties;

/**
 * 功能:Mybatis 攔截器
 * 說明:攔截SQL執(zhí)行前的語句 根據(jù)自定義注解對方法進(jìn)行公司查詢
 *
 * 開發(fā):Bruce.Liu Create by 2018-03-12 20:15
 */

@Component
@Intercepts(
    {
        @Signature(
                type = StatementHandler.class,
                method = "prepare",
                args = {
                        Connection.class ,
                        Integer.class
                }
        )
    }
)
public class CompanySqlInterceptor implements Interceptor {

    @Value("${base.config.companyId}")
    private String COMPANY_ID;

    @Override
    public Object intercept(Invocation invocation) throws Throwable {
        try{
            if(invocation.getTarget() instanceof RoutingStatementHandler){
                RoutingStatementHandler statementHandler = (RoutingStatementHandler)invocation.getTarget();
                StatementHandler delegate = (StatementHandler) ReflectHelper.getFieldValue(statementHandler, "delegate");
                BoundSql boundSql = delegate.getBoundSql();
                MappedStatement mappedStatement = (MappedStatement) ReflectHelper.getFieldValue(delegate, "mappedStatement");
                Class<?> classType = Class.forName(mappedStatement.getId().substring(0,mappedStatement.getId().lastIndexOf(".")));
                String mName  = mappedStatement.getId().substring(mappedStatement.getId().lastIndexOf(".") + 1 ,mappedStatement.getId().length());
                for(Method method : classType.getDeclaredMethods()){
                    if(method.isAnnotationPresent(HmwCompanyAnnotation.class) && mName.equals(method.getName()) )
                    {
                        HmwCompanyAnnotation companyAnnotation =  method.getAnnotation(HmwCompanyAnnotation.class);
                        String sql = boundSql.getSql();
                        if( mappedStatement.getSqlCommandType().toString().equals(Const.SELECT) ||
                               mappedStatement.getSqlCommandType().toString().equals(Const.UPDATE) ||
                                 mappedStatement.getSqlCommandType().toString().equals(Const.DELETE )){
                            if(companyAnnotation.Alias().equals("")){
                                sql = sql + " and COMPANY_ID = '"+ COMPANY_ID +"' ";
                            } else {
                                sql = sql + " and "+ companyAnnotation.Alias()+".COMPANY_ID = '" + COMPANY_ID +"' ";
                            }
                        } else if( mappedStatement.getSqlCommandType().toString().equals(Const.INSERT)){
                          // System.err.println("Insert 實(shí)現(xiàn)已移到Service層處理了");
                        }
                        //System.err.println("執(zhí)行后SQL:"+sql);
                        ReflectHelper.setFieldValue(boundSql, "sql", sql);
                        break;
                    }
                }

            }
        } catch (Exception e){
            e.printStackTrace();
        }
        return invocation.proceed();
    }

    @Override
    public Object plugin(Object target) {
        if (target instanceof StatementHandler) {
            return Plugin.wrap(target,this);
        } else {
            return target;
        }
    }

    @Override
    public void setProperties(Properties properties) {

    }

}
  • 利用MyBatis攔截器@Signature對底層StatementHandler.class的 prepare方法進(jìn)行攔截處理
  • 在執(zhí)行JDBC的SQL腳本之前對sql進(jìn)行統(tǒng)一拼接處理
  • 自定義注解標(biāo)示出需要進(jìn)行攔截處理的方法淹遵,通過反射機(jī)制獲取方法以及SqlCommonType 然后作出相應(yīng)的邏輯處理
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市七问,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 216,402評論 6 499
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件复隆,死亡現(xiàn)場離奇詭異,居然都是意外死亡姆涩,警方通過查閱死者的電腦和手機(jī)挽拂,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,377評論 3 392
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來骨饿,“玉大人亏栈,你說我怎么就攤上這事『曜福” “怎么了绒北?”我有些...
    開封第一講書人閱讀 162,483評論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長察署。 經(jīng)常有香客問我镇饮,道長,這世上最難降的妖魔是什么箕母? 我笑而不...
    開封第一講書人閱讀 58,165評論 1 292
  • 正文 為了忘掉前任储藐,我火速辦了婚禮,結(jié)果婚禮上嘶是,老公的妹妹穿的比我還像新娘钙勃。我一直安慰自己,他們只是感情好聂喇,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,176評論 6 388
  • 文/花漫 我一把揭開白布辖源。 她就那樣靜靜地躺著,像睡著了一般希太。 火紅的嫁衣襯著肌膚如雪克饶。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,146評論 1 297
  • 那天誊辉,我揣著相機(jī)與錄音矾湃,去河邊找鬼。 笑死堕澄,一個(gè)胖子當(dāng)著我的面吹牛邀跃,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播蛙紫,決...
    沈念sama閱讀 40,032評論 3 417
  • 文/蒼蘭香墨 我猛地睜開眼拍屑,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了坑傅?” 一聲冷哼從身側(cè)響起僵驰,我...
    開封第一講書人閱讀 38,896評論 0 274
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后蒜茴,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體星爪,經(jīng)...
    沈念sama閱讀 45,311評論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,536評論 2 332
  • 正文 我和宋清朗相戀三年矮男,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片室谚。...
    茶點(diǎn)故事閱讀 39,696評論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡毡鉴,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出秒赤,到底是詐尸還是另有隱情猪瞬,我是刑警寧澤,帶...
    沈念sama閱讀 35,413評論 5 343
  • 正文 年R本政府宣布入篮,位于F島的核電站陈瘦,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏潮售。R本人自食惡果不足惜痊项,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,008評論 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望酥诽。 院中可真熱鬧鞍泉,春花似錦、人聲如沸肮帐。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,659評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽训枢。三九已至托修,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間恒界,已是汗流浹背睦刃。 一陣腳步聲響...
    開封第一講書人閱讀 32,815評論 1 269
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留十酣,地道東北人眯勾。 一個(gè)月前我還...
    沈念sama閱讀 47,698評論 2 368
  • 正文 我出身青樓,卻偏偏與公主長得像婆誓,于是被迫代替她去往敵國和親吃环。 傳聞我的和親對象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,592評論 2 353

推薦閱讀更多精彩內(nèi)容