亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

MyBatis-plus的五種批量插入方式對(duì)比分析

 更新時(shí)間:2023年06月20日 11:56:16   作者:qq_1797801363  
本文主要介紹了MyBatis-plus的五種批量插入方式對(duì)比分析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧

Mybatis批量插入一直是開(kāi)發(fā)者重點(diǎn)關(guān)注的問(wèn)題,本文列舉了Mybatis的五種插入方式進(jìn)行對(duì)比分析,驗(yàn)證了五種批量插入的方式的優(yōu)先級(jí)。

1 準(zhǔn)備工作

1.1 新建spring項(xiàng)目

略。

1.2 導(dǎo)入pom.xml依賴

<dependency>
? ? <groupId>mysql</groupId>
? ? <artifactId>mysql-connector-java</artifactId>
? ? <scope>runtime</scope>
</dependency>
<!--Mybatis依賴-->
<dependency>
? ? <groupId>org.mybatis.spring.boot</groupId>
? ? <artifactId>mybatis-spring-boot-starter</artifactId>
? ? <version>2.2.2</version>
</dependency>
<!--Mybatis-Plus依賴-->
<dependency>
? ? <groupId>com.baomidou</groupId>
? ? <artifactId>mybatis-plus-boot-starter</artifactId>
? ? <version>3.5.2</version>
</dependency>
<dependency>
? ? <groupId>org.projectlombok</groupId>
? ? <artifactId>lombok</artifactId>
? ? <optional>true</optional>
</dependency>

1.3 配置yml文件

server:
  port: 8080
spring:
  datasource:
    username: mysql用戶名
    password: mysql密碼
    url: jdbc:mysql://localhost:3306/數(shù)據(jù)庫(kù)名字?useUnicode=true&characterEncoding=utf-8&useSSL=false&serverTimezone=UTC
    driver-class-name: com.mysql.cj.jdbc.Driver
mybatis:
  mapper-locations: classpath:mapping/*.xml

1.4 創(chuàng)建插入模型

@Data
public class User {
    private int id;
    private String username;
    private String password;
}

2 測(cè)試

2.1 Mybatis利用For循環(huán)批量插入

1、編寫(xiě)UserService服務(wù)類,測(cè)試一萬(wàn)條數(shù)據(jù)的耗時(shí)情況:

@Service
public class UserService {
    @Resource
    private UserMapper userMapper;
    public void InsertUsers(){
        long start = System.currentTimeMillis();
        for(int i = 0 ;i < 10000; i++) {
            User user = new User();
            user.setUsername("name" + i);
            user.setPassword("password" + i);
            userMapper.insertUsers(user);
        }
        long end = System.currentTimeMillis();
        System.out.println("一萬(wàn)條數(shù)據(jù)總耗時(shí):" + (end-start) + "ms" );
    }
}

2、編寫(xiě)UserMapper接口

@Mapper
public interface UserMapper {
    Integer insertUsers(User user);
}

3、編寫(xiě)UserMapper.xml文件

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapper
        PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
        "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.ithuang.demo.mapper.UserMapper">
    <insert id="insertUsers">
        INSERT INTO user (username, password)
        VALUES(#{username}, #{password})
    </insert>
</mapper>

4、進(jìn)行單元測(cè)試

@SpringBootTest
class DemoApplicationTests {
    @Resource
    private UserService userService;
    @Test
    public void insert(){
        userService.InsertUsers();
    }
}

5、輸出結(jié)果

一萬(wàn)條數(shù)據(jù)耗時(shí)26348ms

2.2 MyBatis的手動(dòng)批量提交

1、其他保持不變,Service層作稍微的變化

@Service
public class UserService {
    @Resource
    private UserMapper userMapper;
    @Resource
    private SqlSessionTemplate sqlSessionTemplate;
    public void InsertUsers(){
        //關(guān)閉自動(dòng)提交
        SqlSession sqlSession = sqlSessionTemplate.getSqlSessionFactory().openSession(ExecutorType.BATCH, false);
        UserMapper userMapper = sqlSession.getMapper(UserMapper.class);
        long start = System.currentTimeMillis();
        for(int i = 0 ;i < 10000; i++) {
            User user = new User();
            user.setUsername("name" + i);
            user.setPassword("password" + i);
            userMapper.insertUsers(user);
        }
        sqlSession.commit();
        long end = System.currentTimeMillis();
        System.out.println("一萬(wàn)條數(shù)據(jù)總耗時(shí):" + (end-start) + "ms" );
    }
}

2、結(jié)果輸出

一萬(wàn)條數(shù)據(jù)總耗時(shí):24516ms。

2.3 Mybatis以集合方式批量新增

1、編寫(xiě)UserService服務(wù)類

@Service
public class UserService {
    @Resource
    private UserMapper userMapper;
    public void InsertUsers(){
        long start = System.currentTimeMillis();
        List<User> userList = new ArrayList<>();
        User user;
        for(int i = 0 ;i < 10000; i++) {
            user = new User();
            user.setUsername("name" + i);
            user.setPassword("password" + i);
            userList.add(user);
        }
        userMapper.insertUsers(userList);
        long end = System.currentTimeMillis();
        System.out.println("一萬(wàn)條數(shù)據(jù)總耗時(shí):" + (end-start) + "ms" );
    }
}

2、編寫(xiě)UserMapper接口

@Mapper
public interface UserMapper {
    Integer insertUsers(List<User> userList);
}

3、編寫(xiě)UserMapper.xml文件

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapper
        PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
        "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.ithuang.demo.mapper.UserMapper">
    <insert id="insertUsers">
        INSERT INTO user (username, password)
        VALUES
        <foreach collection ="userList" item="user" separator =",">
            (#{user.username}, #{user.password})
        </foreach>
    </insert>
</mapper>

4、輸出結(jié)果

一萬(wàn)條數(shù)據(jù)總耗時(shí):521ms

2.4 MyBatis-Plus提供的SaveBatch方法

1、編寫(xiě)UserService服務(wù)

@Service
public class UserService extends ServiceImpl<UserMapper, User> implements IService<User> {
    public void InsertUsers(){
        long start = System.currentTimeMillis();
        List<User> userList = new ArrayList<>();
        User user;
        for(int i = 0 ;i < 10000; i++) {
            user = new User();
            user.setUsername("name" + i);
            user.setPassword("password" + i);
            userList.add(user);
        }
        saveBatch(userList);
        long end = System.currentTimeMillis();
        System.out.println("一萬(wàn)條數(shù)據(jù)總耗時(shí):" + (end-start) + "ms" );
    }
}

2、編寫(xiě)UserMapper接口

@Mapper
public interface UserMapper extends BaseMapper<User> {
}

3、單元測(cè)試結(jié)果

一萬(wàn)條數(shù)據(jù)總耗時(shí):24674ms

2.5 MyBatis-Plus提供的InsertBatchSomeColumn方法

1、編寫(xiě)EasySqlInjector 自定義類

public class EasySqlInjector extends DefaultSqlInjector {
    @Override
    public List<AbstractMethod> getMethodList(Class<?> mapperClass, TableInfo tableInfo) {
        // 注意:此SQL注入器繼承了DefaultSqlInjector(默認(rèn)注入器),調(diào)用了DefaultSqlInjector的getMethodList方法,保留了mybatis-plus的自帶方法
        List<AbstractMethod> methodList = super.getMethodList(mapperClass, tableInfo);
        methodList.add(new InsertBatchSomeColumn(i -> i.getFieldFill() != FieldFill.UPDATE));
        return methodList;
    }
}

2、定義核心配置類注入此Bean

@Configuration
public class MybatisPlusConfig {
@Bean
public EasySqlInjector sqlInjector() {
? ? return new EasySqlInjector();
}
}

3、編寫(xiě)UserService服務(wù)類

public class UserService{
    @Resource
    private UserMapper userMapper;
    public void InsertUsers(){
        long start = System.currentTimeMillis();
        List<User> userList = new ArrayList<>();
        User user;
        for(int i = 0 ;i < 10000; i++) {
            user = new User();
            user.setUsername("name" + i);
            user.setPassword("password" + i);
            userList.add(user);
        }
        userMapper.insertBatchSomeColumn(userList);
        long end = System.currentTimeMillis();
        System.out.println("一萬(wàn)條數(shù)據(jù)總耗時(shí):" + (end-start) + "ms" );
    }
}

4、編寫(xiě)EasyBaseMapper接口

public interface EasyBaseMapper<T> extends BaseMapper<T> {
    /**
     * 批量插入 僅適用于mysql
     *
     * @param entityList 實(shí)體列表
     * @return 影響行數(shù)
     */
    Integer insertBatchSomeColumn(Collection<T> entityList);
}

5、編寫(xiě)UserMapper接口

@Mapper
public interface UserMapper<T> extends EasyBaseMapper<User> {
}

6、單元測(cè)試結(jié)果

一萬(wàn)條數(shù)據(jù)總耗時(shí):575ms

2.6 JDBC原生的批量插入

1、編寫(xiě)JDBC池化工具類

public class JDBCDruidUtils {
? ? private static DataSource dataSource;
? ? private static Connection conn;
? ? /*
? ?創(chuàng)建數(shù)據(jù)Properties集合對(duì)象加載加載配置文件
? ? */
? ? static {
? ? ? ? Properties pro = new Properties();
? ? ? ? //加載數(shù)據(jù)庫(kù)連接池對(duì)象
? ? ? ? try {
? ? ? ? ? ? //獲取數(shù)據(jù)庫(kù)連接池對(duì)象
? ? ? ? ? ? pro.load(JDBCDruidUtils.class.getClassLoader().getResourceAsStream("druid.properties"));
? ? ? ? ? ? dataSource = DruidDataSourceFactory.createDataSource(pro);
? ? ? ? } catch (Exception e) {
? ? ? ? ? ? e.printStackTrace();
? ? ? ? }
? ? }
? ? /*
? ? 獲取連接
? ? ?*/
? ? public static Connection getConnection() throws SQLException {
? ? ? ? return dataSource.getConnection();
? ? }
? ? /**
? ? ?* 關(guān)閉conn,和 statement獨(dú)對(duì)象資源
? ? ?*
? ? ?* @param connection
? ? ?* @param statement
? ? ?* @MethodName: close
? ? ?* @return: void
? ? ?*/
? ? public static void close(Connection connection, Statement statement) {
? ? ? ? if (connection != null) {
? ? ? ? ? ? try {
? ? ? ? ? ? ? ? connection.close();
? ? ? ? ? ? } catch (SQLException e) {
? ? ? ? ? ? ? ? e.printStackTrace();
? ? ? ? ? ? }
? ? ? ? }
? ? ? ? if (statement != null) {
? ? ? ? ? ? try {
? ? ? ? ? ? ? ? statement.close();
? ? ? ? ? ? } catch (SQLException e) {
? ? ? ? ? ? ? ? e.printStackTrace();
? ? ? ? ? ? }
? ? ? ? }
? ? }
? ? /**
? ? ?* 關(guān)閉 conn , statement 和resultset三個(gè)對(duì)象資源
? ? ?*
? ? ?* @param connection
? ? ?* @param statement
? ? ?* @param resultSet
? ? ?* @MethodName: close
? ? ?* @return: void
? ? ?*/
? ? public static void close(Connection connection, Statement statement, ResultSet resultSet) {
? ? ? ? close(connection, statement);
? ? ? ? if (resultSet != null) {
? ? ? ? ? ? try {
? ? ? ? ? ? ? ? resultSet.close();
? ? ? ? ? ? } catch (SQLException e) { ? ? ? ? ? ? e.printStackTrace();
? ? ? ? ? ? }
? ? ? ? }
? ? }
? ? /*
? ? 獲取連接池對(duì)象
? ? ?*/
? ? public static DataSource getDataSource() {
? ? ? ? return dataSource;
? ? }
}
# druid.properties配置
driverClassName=com.mysql.cj.jdbc.Driver
url=jdbc:mysql://localhost:3306/數(shù)據(jù)庫(kù)?useUnicode=true&characterEncoding=utf-8&useSSL=false&serverTimezone=UTC
username=用戶名
password=密碼
initialSize=10
maxActive=50
maxWait=60000

2、編寫(xiě)UserService服務(wù)類

public void InsertUsersByJdbc() {
        long start = System.currentTimeMillis();
        Connection connection = null;
        PreparedStatement ps = null;
        try {
            connection = JDBCDruidUtils.getConnection();
            //控制事務(wù):默認(rèn)不提交
            connection.setAutoCommit(false);
            String sql = "INSERT INTO user (username, password) VALUES (?, ?)";
            ps = connection.prepareStatement(sql);
            User user;
            for (int i = 0; i < 10000; i++) {
                user = new User();
                user.setUsername("name" + i);
                user.setPassword("password" + i);
                ps.setString(1, user.getUsername());
                ps.setString(2, user.getPassword());
                //將一組參數(shù)添加到此 PreparedStatement 對(duì)象的批處理命令中。
                ps.addBatch();
            }
            //執(zhí)行批處理
            ps.executeBatch();
            //手動(dòng)提交事務(wù)
            connection.commit();
        } catch (SQLException e) {
            throw new RuntimeException(e);
        } finally {
            JDBCDruidUtils.close(connection, ps);
        }
        long end = System.currentTimeMillis();
        System.out.println("一萬(wàn)條數(shù)據(jù)總耗時(shí):" + (end - start) + "ms");
    }

3、輸出結(jié)果

1萬(wàn)數(shù)據(jù)總耗時(shí)19000ms。

3 總結(jié)

大量數(shù)據(jù)的場(chǎng)景下性能對(duì)比:InsertBatchSomeColumn>自定義xml以集合的方式>Jdbc原生>SaveBatch>手動(dòng)for循環(huán)批量>自動(dòng)for循環(huán)批量。
網(wǎng)上很多人都說(shuō)JDBC原生性能很好,但是我發(fā)現(xiàn)其非常差,有可能是我使用的是mybatis-plus依賴,如果這是推論正確,那就可以證明mybatis-plus在mybatis的基礎(chǔ)上不僅增強(qiáng)了功能也增強(qiáng)了性能。所以可以得出結(jié)論:開(kāi)發(fā)中用mybatis-plus是沒(méi)有錯(cuò)的,如果想提高性能,只能實(shí)施其他方案,比如分庫(kù)分表,千萬(wàn)別想著JDBC原生性能更好。

到此這篇關(guān)于MyBatis-plus的五種批量插入方式對(duì)比分析的文章就介紹到這了,更多相關(guān)MyBatis-plus 批量插入內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

最新評(píng)論