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

詳解如何使用Mybatis的攔截器

 更新時(shí)間:2024年03月07日 08:21:14   作者:一只愛擼貓的程序猿  
MyBatis?攔截器是?MyBatis?提供的一個(gè)強(qiáng)大特性,它允許你在?MyBatis?執(zhí)行其核心邏輯的關(guān)鍵節(jié)點(diǎn)插入自定義邏輯,從而改變?MyBatis?的默認(rèn)行為,本文給大家詳細(xì)介紹了如何使用Mybatis的攔截器,需要的朋友可以參考下

MyBatis 攔截器是 MyBatis 提供的一個(gè)強(qiáng)大特性,它允許你在 MyBatis 執(zhí)行其核心邏輯的關(guān)鍵節(jié)點(diǎn)插入自定義邏輯,從而改變 MyBatis 的默認(rèn)行為。這類似于面向切面編程(AOP),允許開發(fā)者在不改變?cè)写a的情況下,增強(qiáng)或修改原有功能。

在 MyBatis 中,攔截器可以應(yīng)用于以下四種類型的對(duì)象:

Executor:負(fù)責(zé) MyBatis 中的 SQL 執(zhí)行過程,攔截 Executor 可以在 SQL 執(zhí)行前后添加自定義邏輯。

Executor 的核心職責(zé)

在 MyBatis 中,Executor 接口定義了數(shù)據(jù)庫(kù)操作的核心方法,如 update、query、commitrollback 等。MyBatis 提供了幾種 Executor 的實(shí)現(xiàn),例如 SimpleExecutor、ReuseExecutorBatchExecutor 等,它們各自有不同的特點(diǎn)和用途。

SimpleExecutor 為例,它是最基本的 Executor 實(shí)現(xiàn),每次操作都會(huì)創(chuàng)建一個(gè)新的 Statement 對(duì)象。

源碼解析

以下是 Executor 接口中 query 方法的一個(gè)簡(jiǎn)化版實(shí)現(xiàn),展示了 MyBatis 如何執(zhí)行一個(gè)查詢操作:

public class SimpleExecutor extends BaseExecutor {

    public <E> List<E> query(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler, BoundSql boundSql) {
        // 準(zhǔn)備 SQL
        Statement stmt = null;
        try {
            Configuration configuration = ms.getConfiguration();
            StatementHandler handler = configuration.newStatementHandler(this, ms, parameter, rowBounds, resultHandler, boundSql);
            stmt = prepareStatement(handler, ms.getStatementLog());
            return handler.query(stmt, resultHandler);
        } finally {
            closeStatement(stmt);
        }
    }

    private Statement prepareStatement(StatementHandler handler, Log statementLog) {
        Statement stmt;
        Connection connection = getConnection(statementLog);
        stmt = handler.prepare(connection, transaction.getTimeout());
        handler.parameterize(stmt);
        return stmt;
    }
}

這段代碼展示了查詢操作的基本流程:創(chuàng)建 StatementHandler,準(zhǔn)備和參數(shù)化 Statement,執(zhí)行查詢,最后關(guān)閉 Statement。

攔截器的實(shí)現(xiàn)和應(yīng)用

攔截器允許開發(fā)者在 MyBatis 核心操作執(zhí)行的關(guān)鍵節(jié)點(diǎn)插入自定義邏輯。攔截器需要實(shí)現(xiàn) Interceptor 接口,并定義要攔截的目標(biāo)和方法。

當(dāng) MyBatis 初始化時(shí),它會(huì)檢測(cè)配置中的攔截器,并在執(zhí)行相應(yīng)操作時(shí)調(diào)用這些攔截器。攔截器中的 intercept 方法將在目標(biāo)方法執(zhí)行時(shí)被調(diào)用。

public class ExampleExecutorInterceptor implements Interceptor {

    @Override
    public Object intercept(Invocation invocation) throws Throwable {
        // 在 SQL 執(zhí)行前的邏輯
        System.out.println("Before executing query");

        // 執(zhí)行原有邏輯
        Object returnValue = invocation.proceed();

        // 在 SQL 執(zhí)行后的邏輯
        System.out.println("After executing query");

        return returnValue;
    }
}

在這個(gè)例子中,intercept 方法實(shí)現(xiàn)了在查詢執(zhí)行前后打印消息的邏輯。invocation.proceed() 是關(guān)鍵,它觸發(fā)了原本的操作(如查詢)。通過這種方式,開發(fā)者可以在不修改原有代碼的基礎(chǔ)上增強(qiáng)或改變 MyBatis 的行為。

ParameterHandler:負(fù)責(zé) MyBatis 中的參數(shù)處理,通過攔截 ParameterHandler,可以在 SQL 語句綁定參數(shù)前后添加自定義邏輯。

ParameterHandler 的工作原理

ParameterHandler 的主要職責(zé)是為 SQL 語句綁定正確的參數(shù)值。在執(zhí)行 SQL 之前,MyBatis 會(huì)通過 ParameterHandler 接口的實(shí)現(xiàn)類來遍歷方法傳入的參數(shù),并將它們?cè)O(shè)置到 JDBC 的 PreparedStatement 中。

MyBatis 默認(rèn)提供了 DefaultParameterHandler 類作為 ParameterHandler 接口的實(shí)現(xiàn),用于處理參數(shù)的設(shè)置工作。

源碼解析

下面是一個(gè)簡(jiǎn)化的 ParameterHandler 使用示例,演示了如何在 MyBatis 中處理參數(shù)綁定:

public class DefaultParameterHandler implements ParameterHandler {
    private final MappedStatement mappedStatement;
    private final Object parameterObject;
    private final BoundSql boundSql;
    private final TypeHandlerRegistry typeHandlerRegistry;

    public DefaultParameterHandler(MappedStatement mappedStatement, Object parameterObject, BoundSql boundSql) {
        this.mappedStatement = mappedStatement;
        this.parameterObject = parameterObject;
        this.boundSql = boundSql;
        this.typeHandlerRegistry = mappedStatement.getConfiguration().getTypeHandlerRegistry();
    }

    @Override
    public void setParameters(PreparedStatement ps) {
        ErrorContext.instance().activity("setting parameters").object(mappedStatement.getParameterMap().getId());
        List<ParameterMapping> parameterMappings = boundSql.getParameterMappings();
        if (parameterMappings != null) {
            for (int i = 0; i < parameterMappings.size(); i++) {
                ParameterMapping parameterMapping = parameterMappings.get(i);
                if (parameterMapping.getMode() != ParameterMode.OUT) {
                    Object value;
                    String propertyName = parameterMapping.getProperty();
                    if (boundSql.hasAdditionalParameter(propertyName)) {
                        value = boundSql.getAdditionalParameter(propertyName);
                    } else if (parameterObject == null) {
                        value = null;
                    } else if (typeHandlerRegistry.hasTypeHandler(parameterObject.getClass())) {
                        value = parameterObject;
                    } else {
                        MetaObject metaObject = mappedStatement.getConfiguration().newMetaObject(parameterObject);
                        value = metaObject.getValue(propertyName);
                    }
                    TypeHandler typeHandler = parameterMapping.getTypeHandler();
                    JdbcType jdbcType = parameterMapping.getJdbcType();
                    if (value == null && jdbcType == null) {
                        jdbcType = configuration.getJdbcTypeForNull();
                    }
                    typeHandler.setParameter(ps, i + 1, value, jdbcType);
                }
            }
        }
    }
}

在上述代碼中,setParameters 方法是核心,它負(fù)責(zé)將參數(shù)值綁定到 PreparedStatement。首先,它會(huì)遍歷所有的 ParameterMapping,這些映射信息定義了如何從參數(shù)對(duì)象中獲取具體的值。然后,它使用相應(yīng)的 TypeHandler 來處理參數(shù)值的類型轉(zhuǎn)換,并將轉(zhuǎn)換后的值設(shè)置到 PreparedStatement 中。

攔截器的應(yīng)用

通過攔截 ParameterHandlersetParameters 方法,可以在參數(shù)綁定前后插入自定義邏輯。例如,可以在參數(shù)綁定之前對(duì)參數(shù)進(jìn)行日志記錄,或者對(duì)參數(shù)值進(jìn)行額外的處理。

@Intercepts({
    @Signature(type = ParameterHandler.class, method = "setParameters", args = PreparedStatement.class)
})
public class ExampleParameterHandlerInterceptor implements Interceptor {
    @Override
    public Object intercept(Invocation invocation) throws Throwable {
        ParameterHandler parameterHandler = (ParameterHandler) invocation.getTarget();
        // 在參數(shù)綁定前的自定義邏輯
        System.out.println("Before parameter binding");

        // 繼續(xù)執(zhí)行原邏輯
        Object result = invocation.proceed();

        // 在參數(shù)綁定后的自定義邏輯
        System.out.println("After parameter binding");

        return result;
    }
}

在這個(gè)例子中,intercept 方法會(huì)在 setParameters 被調(diào)用時(shí)執(zhí)行,允許開發(fā)者在參數(shù)被綁定到 SQL 語句之前和之后執(zhí)行自定義代碼。

ResultSetHandler:負(fù)責(zé)處理 JDBC 返回的 ResultSet 結(jié)果集,攔截 ResultSetHandler 可以在結(jié)果集處理前后添加自定義邏輯。

ResultSetHandler 的工作原理

當(dāng) MyBatis 執(zhí)行查詢操作后,會(huì)得到一個(gè) ResultSet,ResultSetHandler 的職責(zé)就是遍歷這個(gè) ResultSet,并將其行轉(zhuǎn)換為 Java 對(duì)象。MyBatis 中默認(rèn)的 ResultSetHandler 實(shí)現(xiàn)是 DefaultResultSetHandler

源碼解析

下面是 DefaultResultSetHandler 處理結(jié)果集的一個(gè)簡(jiǎn)化版示例,幫助理解其工作機(jī)制:

public class DefaultResultSetHandler implements ResultSetHandler {

    private final MappedStatement mappedStatement;
    private final RowBounds rowBounds;

    public DefaultResultSetHandler(MappedStatement mappedStatement, RowBounds rowBounds) {
        this.mappedStatement = mappedStatement;
        this.rowBounds = rowBounds;
    }

    @Override
    public <E> List<E> handleResultSets(Statement stmt) throws SQLException {
        ResultSet rs = stmt.getResultSet();
        List<E> resultList = new ArrayList<>();
        while (rs.next()) {
            E resultObject = getRowValue(rs);
            resultList.add(resultObject);
        }
        return resultList;
    }

    private <E> E getRowValue(ResultSet rs) throws SQLException {
        // 實(shí)際的結(jié)果對(duì)象映射邏輯
        // 這里通常會(huì)涉及到 ResultMap 的處理
        return ...;
    }
}

在這個(gè)簡(jiǎn)化的例子中,handleResultSets 方法遍歷 ResultSet,對(duì)每一行調(diào)用 getRowValue 方法將其轉(zhuǎn)換為一個(gè) Java 對(duì)象,最終返回一個(gè)對(duì)象列表。

攔截器的應(yīng)用

通過攔截 ResultSetHandlerhandleResultSets 方法,開發(fā)者可以在結(jié)果集被處理成 Java 對(duì)象前后插入自定義邏輯。這可以用于額外的結(jié)果處理,比如對(duì)查詢結(jié)果的后處理或?qū)徲?jì)日志記錄等。

@Intercepts({
    @Signature(type = ResultSetHandler.class, method = "handleResultSets", args = Statement.class)
})
public class ExampleResultSetHandlerInterceptor implements Interceptor {
    @Override
    public Object intercept(Invocation invocation) throws Throwable {
        // 在結(jié)果集處理前的邏輯
        System.out.println("Before handling result sets");

        // 執(zhí)行原有邏輯
        Object result = invocation.proceed();

        // 在結(jié)果集處理后的邏輯
        System.out.println("After handling result sets");

        return result;
    }
}

在這個(gè)示例中,intercept 方法在 handleResultSets 被調(diào)用時(shí)執(zhí)行,允許在結(jié)果集轉(zhuǎn)換為 Java 對(duì)象的前后執(zhí)行自定義代碼。

StatementHandler:負(fù)責(zé)對(duì) JDBC Statement 的操作,通過攔截 StatementHandler,可以在 SQL 語句被執(zhí)行前后添加自定義邏輯。

StatementHandler 的工作原理

StatementHandler 主要有三個(gè)實(shí)現(xiàn)類:SimpleStatementHandler、PreparedStatementHandlerCallableStatementHandler,分別對(duì)應(yīng)于 JDBC 的 Statement、PreparedStatementCallableStatement。這些類處理 SQL 的不同執(zhí)行方式,其中 PreparedStatementHandler 是最常用的,因?yàn)樗С謪?shù)化的 SQL 語句,有助于提高性能和安全性。

源碼解析

以下是 StatementHandler 接口的一個(gè)簡(jiǎn)化版實(shí)現(xiàn)(PreparedStatementHandler),演示了 MyBatis 如何準(zhǔn)備和執(zhí)行 SQL 語句:

public class PreparedStatementHandler implements StatementHandler {

    private final MappedStatement mappedStatement;
    private final Object parameterObject;
    private final BoundSql boundSql;
    private final ParameterHandler parameterHandler;

    public PreparedStatementHandler(MappedStatement mappedStatement, Object parameterObject, BoundSql boundSql) {
        this.mappedStatement = mappedStatement;
        this.parameterObject = parameterObject;
        this.boundSql = boundSql;
        this.parameterHandler = mappedStatement.getConfiguration().newParameterHandler(mappedStatement, parameterObject, boundSql);
    }

    @Override
    public PreparedStatement prepare(Connection connection) throws SQLException {
        String sql = boundSql.getSql();
        PreparedStatement pstmt = connection.prepareStatement(sql);
        parameterHandler.setParameters(pstmt);
        return pstmt;
    }

    @Override
    public int update(PreparedStatement pstmt) throws SQLException {
        pstmt.execute();
        return pstmt.getUpdateCount();
    }

    @Override
    public <E> List<E> query(PreparedStatement pstmt, ResultHandler resultHandler) throws SQLException {
        pstmt.execute();
        return resultSetHandler.handleResultSets(pstmt);
    }
}

在這個(gè)簡(jiǎn)化的實(shí)現(xiàn)中,prepare 方法負(fù)責(zé)創(chuàng)建 PreparedStatement 并通過 ParameterHandler 設(shè)置參數(shù)。updatequery 方法則用于執(zhí)行 SQL 語句并處理執(zhí)行結(jié)果。

攔截器的應(yīng)用

通過攔截 StatementHandler 的方法,可以在 SQL 語句執(zhí)行的關(guān)鍵節(jié)點(diǎn)加入自定義邏輯。例如,可以攔截 prepare 方法,在 SQL 語句被準(zhǔn)備之前或之后添加邏輯:

@Intercepts({
    @Signature(type = StatementHandler.class, method = "prepare", args = {Connection.class, Integer.class})
})
public class ExampleStatementHandlerInterceptor implements Interceptor {
    @Override
    public Object intercept(Invocation invocation) throws Throwable {
        // 在 SQL 語句準(zhǔn)備前的邏輯
        System.out.println("Before preparing the SQL statement");

        // 執(zhí)行原有邏輯
        Object result = invocation.proceed();

        // 在 SQL 語句準(zhǔn)備后的邏輯
        System.out.println("After preparing the SQL statement");

        return result;
    }
}

在這個(gè)示例中,intercept 方法會(huì)在 prepare 方法執(zhí)行時(shí)被調(diào)用,允許開發(fā)者在 SQL 語句被準(zhǔn)備和執(zhí)行前后插入自定義代碼。

實(shí)際案例

讓我們?cè)谝粋€(gè) Spring Boot 項(xiàng)目中結(jié)合 MyBatis 的 Executor、ParameterHandlerResultSetHandler、和 StatementHandler 構(gòu)建一個(gè)實(shí)際案例。我們將創(chuàng)建一個(gè)簡(jiǎn)單的用戶管理系統(tǒng),其中包含用戶的添加、查詢功能,并通過攔截器在關(guān)鍵節(jié)點(diǎn)添加日志記錄、性能監(jiān)控等自定義邏輯。

步驟 1: 定義實(shí)體和映射文件

首先,定義一個(gè) User 實(shí)體:

public class User {
    private Integer id;
    private String name;
    private String email;

    // Getters and setters...
}

然后,創(chuàng)建一個(gè) MyBatis 映射文件 UserMapper.xml

<mapper namespace="com.example.mybatisdemo.mapper.UserMapper">
    <insert id="insertUser" parameterType="User">
        INSERT INTO users (name, email) VALUES (#{name}, #{email})
    </insert>
    
    <select id="getUserById" parameterType="int" resultType="User">
        SELECT * FROM users WHERE id = #{id}
    </select>
</mapper>

步驟 2: 定義 Mapper 接口

@Mapper
public interface UserMapper {
    void insertUser(User user);
    User getUserById(int id);
}

步驟 3: 定義攔截器

  • Executor 攔截器 - 記錄執(zhí)行時(shí)間:
@Intercepts({
    @Signature(type = Executor.class, method = "update", args = {MappedStatement.class, Object.class}),
    @Signature(type = Executor.class, method = "query", args = {MappedStatement.class, Object.class, RowBounds.class, ResultHandler.class})
})
public class ExecutorInterceptor implements Interceptor {
    @Override
    public Object intercept(Invocation invocation) throws Throwable {
        long startTime = System.currentTimeMillis();
        Object result = invocation.proceed();
        long endTime = System.currentTimeMillis();
        System.out.println("Execution time: " + (endTime - startTime) + "ms");
        return result;
    }
}
  • ParameterHandler 攔截器 - 記錄參數(shù)信息:
@Intercepts({
    @Signature(type = ParameterHandler.class, method = "setParameters", args = {PreparedStatement.class})
})
public class ParameterHandlerInterceptor implements Interceptor {
    @Override
    public Object intercept(Invocation invocation) throws Throwable {
        ParameterHandler parameterHandler = (ParameterHandler) invocation.getTarget();
        System.out.println("Parameters: " + parameterHandler.getParameterObject());
        return invocation.proceed();
    }
}
  • ResultSetHandler 攔截器 - 在結(jié)果集后打印日志:
@Intercepts({
    @Signature(type = ResultSetHandler.class, method = "handleResultSets", args = {Statement.class})
})
public class ResultSetHandlerInterceptor implements Interceptor {
    @Override
    public Object intercept(Invocation invocation) throws Throwable {
        Object result = invocation.proceed();
        System.out.println("Result: " + result);
        return result;
    }
}
  • StatementHandler 攔截器 - 修改 SQL 語句:
@Intercepts({
    @Signature(type = StatementHandler.class, method = "prepare", args = {Connection.class, Integer.class})
})
public class StatementHandlerInterceptor implements Interceptor {
    @Override
    public Object intercept(Invocation invocation) throws Throwable {
        StatementHandler handler = (StatementHandler) invocation.getTarget();
        BoundSql boundSql = handler.getBoundSql();
        String sql = boundSql.getSql();
        System.out.println("Original SQL: " + sql);
        
        // 此處可根據(jù)需要修改 sql
        // String modifiedSql = sql.replace(...);

        return invocation.proceed();
    }
}

步驟 4: 注冊(cè)攔截器

在 Spring Boot 配置類中注冊(cè)攔截器:

@Configuration
public class MyBatisConfig {
    
    @Bean
    public ExecutorInterceptor executorInterceptor() {
        return new ExecutorInterceptor();
    }

    @Bean
    public ParameterHandlerInterceptor parameterHandlerInterceptor() {
        return new ParameterHandlerInterceptor();
    }

    @Bean
    public ResultSetHandlerInterceptor resultSetHandlerInterceptor() {
        return new ResultSetHandlerInterceptor();
    }

    @Bean
    public StatementHandlerInterceptor statementHandlerInterceptor() {
        return new StatementHandlerInterceptor();
    }
}

步驟 5: 使用 Mapper 進(jìn)行操作

在你的服務(wù)層或控制器中,使用 UserMapper 來執(zhí)行

數(shù)據(jù)庫(kù)操作:

@Service
public class UserService {
    
    @Autowired
    private UserMapper userMapper;

    public void createUser(User user) {
        userMapper.insertUser(user);
    }

    public User getUser(int id) {
        return userMapper.getUserById(id);
    }
}

當(dāng)你執(zhí)行 createUser 或 getUser 方法時(shí),你的攔截器將會(huì)被觸發(fā),你可以看到控制臺(tái)上打印的相關(guān)信息,以及 MyBatis 如何處理 SQL 操作以及如何通過攔截器介入這些過程。

以上就是詳解如何使用Mybatis的攔截器的詳細(xì)內(nèi)容,更多關(guān)于Mybatis攔截器使用的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • 詳解java NIO之Channel(通道)

    詳解java NIO之Channel(通道)

    這篇文章主要介紹了詳解java NIO之Channel(通道)的相關(guān)資料,文中講解非常詳細(xì),示例代碼幫助大家更好的理解和學(xué)習(xí),感興趣的朋友可以了解下
    2020-07-07
  • Java用文件流下載網(wǎng)絡(luò)文件示例代碼

    Java用文件流下載網(wǎng)絡(luò)文件示例代碼

    這篇文章主要介紹了Java用文件流的方式下載網(wǎng)絡(luò)文件,大家參考使用吧
    2013-11-11
  • 基于spring mvc請(qǐng)求controller訪問方式

    基于spring mvc請(qǐng)求controller訪問方式

    這篇文章主要介紹了spring mvc請(qǐng)求controller訪問方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-09-09
  • 解決SpringBoot2多線程無法注入的問題

    解決SpringBoot2多線程無法注入的問題

    這篇文章主要介紹了解決SpringBoot2多線程無法注入的問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧
    2020-08-08
  • 快速學(xué)會(huì)Dubbo的配置環(huán)境及相關(guān)配置

    快速學(xué)會(huì)Dubbo的配置環(huán)境及相關(guān)配置

    本文主要講解Dubbo的環(huán)境與配置,文中運(yùn)用大量代碼和圖片講解的非常詳細(xì),需要學(xué)習(xí)或用到相關(guān)知識(shí)的小伙伴可以參考這篇文章
    2021-09-09
  • JavaWeb實(shí)現(xiàn)文件上傳與下載的方法

    JavaWeb實(shí)現(xiàn)文件上傳與下載的方法

    這篇文章主要介紹了JavaWeb實(shí)現(xiàn)文件上傳與下載的方法的相關(guān)資料,需要的朋友可以參考下
    2016-01-01
  • spring cloud 配置中心native配置方式

    spring cloud 配置中心native配置方式

    這篇文章主要介紹了spring cloud 配置中心native配置方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-09-09
  • Java接口冪等性設(shè)計(jì)原理解析

    Java接口冪等性設(shè)計(jì)原理解析

    這篇文章主要介紹了Java接口冪等性設(shè)計(jì)原理解析,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-05-05
  • 詳解java設(shè)計(jì)模式中的門面模式

    詳解java設(shè)計(jì)模式中的門面模式

    門面模式又叫外觀模式(Facade?Pattern),主要用于隱藏系統(tǒng)的復(fù)雜性,并向客戶端提供了一個(gè)客戶端可以訪問系統(tǒng)的接口,本文通過實(shí)例代碼給大家介紹下java門面模式的相關(guān)知識(shí),感興趣的朋友一起看看吧
    2022-02-02
  • Java中的Comparable和Comparator接口

    Java中的Comparable和Comparator接口

    這篇文章主要介紹了Java中的Comparable和Comparator接口,文章圍繞主題展開詳細(xì)的內(nèi)容戒殺,具有一定的參考價(jià)值,需要的小伙伴可以參考一下
    2022-09-09

最新評(píng)論