All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.jn.sqlhelper.mybatis.batch.MybatisBatchUpdaters Maven / Gradle / Ivy

/*
 * Copyright 2019 the original author or authors.
 *
 * Licensed under the LGPL, Version 3.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at  http://www.gnu.org/licenses/lgpl-3.0.html
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package com.jn.sqlhelper.mybatis.batch;

import com.jn.langx.annotation.NonNull;
import com.jn.langx.annotation.Nullable;
import com.jn.langx.text.StringTemplates;
import com.jn.langx.util.*;
import com.jn.langx.util.collection.Collects;
import com.jn.langx.util.concurrent.completion.CompletableFuture;
import com.jn.langx.util.function.Consumer2;
import com.jn.langx.util.function.Function;
import com.jn.langx.util.function.Supplier;
import com.jn.langx.util.function.Supplier0;
import com.jn.sqlhelper.common.batch.BatchMode;
import com.jn.sqlhelper.common.batch.BatchResult;
import com.jn.sqlhelper.dialect.Dialect;
import com.jn.sqlhelper.dialect.DialectRegistry;
import com.jn.sqlhelper.mybatis.MybatisUtils;
import org.apache.ibatis.session.Configuration;
import org.apache.ibatis.session.SqlSession;
import org.apache.ibatis.session.SqlSessionFactory;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.sql.Connection;
import java.sql.SQLException;
import java.util.List;
import java.util.concurrent.ExecutorService;

public class MybatisBatchUpdaters {
    private static final Logger logger = LoggerFactory.getLogger(MybatisBatchUpdaters.class);

    public static  MybatisBatchUpdater createSimpleBatchUpdater(@NonNull SqlSessionFactory sessionFactory) {
        return createBatchUpdater(sessionFactory, BatchMode.SIMPLE);
    }

    public static  MybatisBatchUpdater createBatchSqlBatchUpdater(@NonNull SqlSessionFactory sessionFactory) {
        return createBatchUpdater(sessionFactory, BatchMode.BATCH_SQL);
    }

    public static  MybatisBatchUpdater createJdbcBatchUpdater(@NonNull SqlSessionFactory sessionFactory) {
        return createBatchUpdater(sessionFactory, BatchMode.JDBC_BATCH);
    }

    public static  MybatisBatchUpdater createBatchUpdater(@NonNull SqlSessionFactory sessionFactory, @Nullable BatchMode batchType) {
        MybatisBatchUpdater updater = null;
        if (batchType != null) {
            switch (batchType) {
                case SIMPLE:
                    updater = new SimpleBatchUpdater();
                    break;
                case BATCH_SQL:
                    updater = new BatchSqlBatchUpdater();
                    break;
                case JDBC_BATCH:
                    updater = new JdbcBatchUpdater();
                    break;
                default:
                    break;
            }
        }
        if (Objs.isNotNull(updater)) {
            updater.setSessionFactory(sessionFactory);
        }
        return updater;
    }

    public static  BatchResult batchUpdate(@NonNull SqlSessionFactory sessionFactory,
                                                 @NonNull String statementIdFQN,
                                                 @Nullable BatchMode batchMode,
                                                 List entities) throws SQLException {
        return batchUpdate(sessionFactory, statementIdFQN, batchMode, entities, 100, null);
    }


    public static  BatchResult batchUpdate(@NonNull SqlSessionFactory sessionFactory,
                                                 @NonNull String statementIdFQN,
                                                 @Nullable BatchMode batchMode,
                                                 List entities,
                                                 int batchSize,
                                                 ExecutorService executor) throws SQLException {
        return batchUpdate(sessionFactory, batchMode, new MybatisBatchStatement(batchMode, statementIdFQN), entities, batchSize, executor);
    }

    public static  BatchResult batchUpdate(@NonNull SqlSessionFactory sessionFactory,
                                                 @NonNull Class mapperClass,
                                                 @NonNull String statementId,
                                                 @Nullable BatchMode batchMode,
                                                 List entities) throws SQLException {
        return batchUpdate(sessionFactory, mapperClass, statementId, batchMode, entities, 100, null);
    }


    public static  BatchResult batchUpdate(@NonNull SqlSessionFactory sessionFactory,
                                                 @NonNull Class mapperClass,
                                                 @NonNull String statementId,
                                                 @Nullable BatchMode batchMode,
                                                 List entities,
                                                 int batchSize,
                                                 @Nullable ExecutorService executor) throws SQLException {
        return batchUpdate(sessionFactory, batchMode, new MybatisBatchStatement(batchMode, mapperClass, statementId), entities, batchSize, executor);
    }

    /**
     * 提供批量更新数据的逻辑。
     * 当最终计算的批次大于1时,那么多个批次会串行执行。
     * 当期望不在当前线程执行时,请传递executor参数
     *
     * @param sessionFactory
     * @param batchMode
     * @param statement
     * @param entities       批量更新的数据
     * @param batchSize      批大小
     * @param executor       异步任务执行器,若存在该参数,则进行异步执行,若不存在该参数,则在当前线程执行。前提是 parallel = false
     * @return
     * @throws SQLException
     */
    public static  BatchResult batchUpdate(@NonNull final SqlSessionFactory sessionFactory,
                                                 @Nullable final BatchMode batchMode,
                                                 @NonNull final MybatisBatchStatement statement,
                                                 List entities,
                                                 int batchSize,
                                                 @Nullable ExecutorService executor) throws SQLException {
        return batchUpdate(sessionFactory, batchMode, statement, entities, batchSize, executor, false);
    }

    /**
     * 提供批量更新数据的逻辑。
     * 当最终计算的批次大于1时,那么多个批次会串行执行。
     * 当期望不在当前线程执行时,请传递executor参数
     *
     * @param sessionFactory
     * @param batchMode
     * @param statement
     * @param entities       批量更新的数据
     * @param batchSize      批大小
     * @param executor       异步任务执行器,若存在该参数,则进行异步执行,若不存在该参数,则在当前线程执行。前提是 parallel = false
     * @param parallel       当数据量被分为多个批次时,是否采用并行执行方式,若采用并行执行方式时,必然不在当前线程执行批量更新任务
     * @return
     * @throws SQLException
     */
    public static  BatchResult batchUpdate(@NonNull final SqlSessionFactory sessionFactory,
                                                 @Nullable final BatchMode batchMode,
                                                 @NonNull final MybatisBatchStatement statement,
                                                 List entities,
                                                 int batchSize,
                                                 @Nullable ExecutorService executor,
                                                 boolean parallel) throws SQLException {

        batchSize = batchSize < 1 ? 100 : batchSize;
        final BatchResult result = new BatchResult();
        result.setStatement(statement);
        result.setParameters(entities);
        result.setRowsAffected(0);
        if (Emptys.isAnyEmpty(entities, statement, sessionFactory)) {
            return result;
        }

        final List> entitiesList = Collects.partitionBySize(entities, batchSize);

        CompletableFuture future = null;

        if (!parallel) {
            if (executor == null) {
                future = CompletableFuture.completedFuture(internalInvokeBatch(sessionFactory, batchMode, statement, entitiesList))
                        .whenComplete(new Consumer2, Throwable>() {
                            @Override
                            public void accept(BatchResult batchResult0, Throwable throwable) {
                                if (batchResult0 != null) {
                                    result.setRowsAffected(result.getRowsAffected() + batchResult0.getRowsAffected());
                                    result.getThrowables().addAll(batchResult0.getThrowables());
                                }
                                if (throwable != null) {
                                    result.getThrowables().add(throwable);
                                }
                            }
                        });
            } else {
                future = CompletableFuture.supplyAsync(new Supplier0>() {
                    @Override
                    public BatchResult get() {
                        try {
                            return internalInvokeBatch(sessionFactory, batchMode, statement, entitiesList);
                        } catch (SQLException e) {
                            throw Throwables.wrapAsRuntimeException(e);
                        }
                    }
                }, executor).whenCompleteAsync(new Consumer2, Throwable>() {
                    @Override
                    public void accept(BatchResult batchResult0, Throwable throwable) {
                        if (batchResult0 != null) {
                            result.setRowsAffected(result.getRowsAffected() + batchResult0.getRowsAffected());
                            result.getThrowables().addAll(batchResult0.getThrowables());
                        }
                        if (throwable != null) {
                            result.getThrowables().add(throwable);
                        }
                    }
                });
            }
        } else {
            List>> futures = Collects.emptyArrayList();
            for (List es : entitiesList) {

                final List segment = es;
                futures.add(
                        (executor == null ? CompletableFuture.supplyAsync(new Supplier0>() {
                            @Override
                            public BatchResult get() {
                                try {
                                    return batch(sessionFactory, batchMode, statement, segment);
                                } catch (SQLException e) {
                                    throw Throwables.wrapAsRuntimeException(e);
                                }
                            }
                        }) : CompletableFuture.supplyAsync(new Supplier0>() {
                            @Override
                            public BatchResult get() {
                                try {
                                    return batch(sessionFactory, batchMode, statement, segment);
                                } catch (SQLException e) {
                                    throw Throwables.wrapAsRuntimeException(e);
                                }
                            }
                        }, executor))
                                .exceptionally(new Function>() {
                                    @Override
                                    public BatchResult apply(Throwable throwable) {
                                        BatchResult result = new BatchResult();
                                        result.setParameters(segment);
                                        result.setRowsAffected(0);
                                        result.setStatement(statement);
                                        result.setThrowables(Collects.newArrayList(throwable));
                                        return result;
                                    }
                                })
                                .whenCompleteAsync(new Consumer2, Throwable>() {
                                    @Override
                                    public void accept(BatchResult batchResult0, Throwable throwable) {
                                        if (batchResult0 != null) {
                                            result.setRowsAffected(result.getRowsAffected() + batchResult0.getRowsAffected());
                                            result.getThrowables().addAll(batchResult0.getThrowables());
                                        }
                                        if (throwable != null) {
                                            result.getThrowables().add(throwable);
                                        }
                                    }
                                }));
            }

            future = CompletableFuture.allOf(futures.toArray(new CompletableFuture[futures.size()]));
        }
        if (future != null) {
            future.join();
        }
        return result;
    }

    /**
     * 串行执行
     * @param sessionFactory
     * @param batchMode
     * @param statement
     * @param entitiesList
     * @param 
     * @return
     * @throws SQLException
     */
    private static  BatchResult internalInvokeBatch(@NonNull final SqlSessionFactory sessionFactory,
                                                          @Nullable final BatchMode batchMode,
                                                          @NonNull final MybatisBatchStatement statement,
                                                          List> entitiesList) throws SQLException {
        BatchResult result = new BatchResult();
        for (List entities : entitiesList) {
            try {
                BatchResult segmentResult = batch(sessionFactory, batchMode, statement, entities);
                if (segmentResult != null) {
                    result.setRowsAffected(result.getRowsAffected() + segmentResult.getRowsAffected());
                    result.getThrowables().addAll(segmentResult.getThrowables());
                }
                if (Emptys.isNotEmpty(result.getThrowables())) {
                    result.getThrowables().addAll(result.getThrowables());
                }
            } catch (Throwable ex) {
                result.addThrowable(ex);
            }

        }
        return result;
    }


    /**
     * @param sessionFactory 这个sessionFactory 不能是 DynamicSqlSessionFactory, 如果是你拿到的是DynamicSqlSessionFactory,可以基于 SqlSessionFactoryProvider来进行帮忙获取真实的session factory
     * @param batchMode
     * @param statement
     * @param entities
     * @param 
     * @return
     * @throws SQLException
     */
    private static  BatchResult batch(@NonNull SqlSessionFactory sessionFactory,
                                            @Nullable BatchMode batchMode,
                                            final MybatisBatchStatement statement,
                                            List entities) throws SQLException {
        Preconditions.checkArgument(Emptys.isNotEmpty(entities));

        // build batch statement
        Preconditions.checkArgument(hasStatement(sessionFactory, statement), new Supplier() {
            @Override
            public String get(Object[] objects) {
                return StringTemplates.formatWithPlaceholder("The statement {} is not exists", statement.getSql());
            }
        });

        if (batchMode != null) {
            return MybatisBatchUpdaters.createBatchUpdater(sessionFactory, batchMode).batchUpdate(statement, entities);
        }

        // find dialect
        Configuration configuration = sessionFactory.getConfiguration();
        String databaseId = configuration.getDatabaseId();
        Dialect dialect = null;
        if (!Strings.isEmpty(databaseId)) {
            dialect = DialectRegistry.getInstance().getDialectByName(databaseId);
        }
        if (Objs.isNull(dialect)) {
            SqlSession session = sessionFactory.openSession();
            Connection connection = session.getConnection();
            dialect = DialectRegistry.getInstance().getDialectByDatabaseMetadata(connection.getMetaData());
            session.close();
        }


        boolean supportsBatchSqlMode = dialect != null && dialect.isSupportsBatchSql();
        MybatisBatchUpdater updater = null;
        BatchResult result = null;
        if (supportsBatchSqlMode) {
            statement.setBatchMode(BatchMode.BATCH_SQL);
            updater = createBatchSqlBatchUpdater(sessionFactory);
            result = updater.batchUpdate(statement, entities);
            if (!result.hasThrowable()) {
                return result;
            }
            logger.warn("Error when execute batch update based on database's batch sql, may be the statement {} not a batch sql, will use jdbc batch method execute it. error: {}", statement.getSql(), result.getThrowables().get(0));
        }
        boolean supportsJdbcBatch = dialect != null && dialect.isSupportsBatchUpdates();
        if (supportsJdbcBatch) {
            statement.setBatchMode(BatchMode.JDBC_BATCH);
            updater = createJdbcBatchUpdater(sessionFactory);
            result = updater.batchUpdate(statement, entities);
            if (result.hasThrowable()) {
                logger.warn("Error when execute batch update based jdbc batch mode, statement: {}, errors:", statement.getSql());
                Collects.forEach(result.getThrowables(), new Consumer2() {
                    @Override
                    public void accept(Integer index, Throwable throwable) {
                        logger.warn("errors[{}]", index, throwable);
                    }
                });
            } else {
                return result;
            }
        }
        statement.setBatchMode(BatchMode.SIMPLE);
        updater = createSimpleBatchUpdater(sessionFactory);
        result = updater.batchUpdate(statement, entities);
        if (result.hasThrowable()) {
            logger.warn("Error when execute batch update based simple batch mode, statement: {}, errors", statement.getSql());
            Collects.forEach(result.getThrowables(), new Consumer2() {
                @Override
                public void accept(Integer index, Throwable throwable) {
                    logger.warn("errors[{}]", index, throwable);
                }
            });
        }
        return result;
    }

    public static boolean hasStatement(@NonNull SqlSessionFactory sessionFactory, @NonNull MybatisBatchStatement statement) {
        return MybatisUtils.hasStatement(sessionFactory, statement.getSql());
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy