【JDBC核心】批量插入
批量插入
批量执行 SQL 语句
当需要成批插入或者更新记录时,可以采用 Java 的批量更新机制,这一机制允许多条语句一次性提交给数据库批量处理。通常情况下比单独提交处理更有效率。
JDBC 的批量处理语句包括下面三个方法:
- addBatch(String):添加需要批量处理的 SQL 语句或是参数;
- executeBatch():执行批量处理语句;
- clearBatch():清空缓存的数据。
通常会遇到两种批量执行 SQL 语句的情况:
- 多条 SQL 语句的批量处理;
- 一个 SQL 语句的批量传参。
高效的批量插入
package cn.parzulpan.jdbc.ch05;
import cn.parzulpan.jdbc.util.JDBCUtils;
import org.junit.Test;
import java.sql.Connection;
import java.sql.PreparedStatement;
import java.sql.Statement;
/**
 * @Author : parzulpan
 * @Time : 2020-12-01
 * @Desc : 批量插入
 */
public class BulkInsertTest {
    // 实现方式一:使用 Statement
    @Test
    public void test1() throws Exception {
        long start = System.currentTimeMillis();
        Connection connection = JDBCUtils.getConnection();
        Statement statement = connection.createStatement();
        for (int i = 0; i < 1000; i++) {
            String sql = "insert into goods(name)values('name_" + i + "')";
            statement.executeUpdate(sql);
        }
        JDBCUtils.closeResource(connection, statement);
        long end = System.currentTimeMillis();
        System.out.println(end - start);    // 1000 -> 131751ms
    }
    // 实现方式二:使用 PrepareStatement
    @Test
    public void test2() throws Exception {
        long start = System.currentTimeMillis();
        Connection connection = JDBCUtils.getConnection();
        String sql = "insert into goods(name)values(?)";
        PreparedStatement statement = connection.prepareStatement(sql);
        for (int i = 0; i < 1000; i++) {
            statement.setString(1, "name_" + i);
            statement.executeUpdate();
        }
        JDBCUtils.closeResource(connection, statement);
        long end = System.currentTimeMillis();
        System.out.println(end - start);    // 1000 -> 122340ms
    }
    // 实现方式三:使用 批量处理语句
    // mysql 服务器默认是关闭批处理的,需要通过一个参数,让 mysql 开启批处理的支持。
    // ?rewriteBatchedStatements=true 写在配置文件的 url 后面
    @Test
    public void test3() throws Exception {
        long start = System.currentTimeMillis();
        Connection connection = JDBCUtils.getConnection();
        String sql = "insert into goods(name)values(?)";
        PreparedStatement statement = connection.prepareStatement(sql);
        for (int i = 0; i < 1000; i++) {
            statement.setString(1, "name_" + i);
            // 1. "攒" sql
            statement.addBatch();
            if (i % 500 == 0) {
                // 2. 执行
                statement.executeBatch();
                // 3. 清空
                statement.clearBatch();
            }
        }
        JDBCUtils.closeResource(connection, statement);
        long end = System.currentTimeMillis();
        System.out.println(end - start);    // 1000 -> 69084ms
    }
    // 实现方式四:设置为不自动提交数据
    @Test
    public void test4() throws Exception {
        long start = System.currentTimeMillis();
        Connection connection = JDBCUtils.getConnection();
        // 1. 设置为不自动提交数据
        connection.setAutoCommit(false);
        String sql = "insert into goods(name)values(?)";
        PreparedStatement statement = connection.prepareStatement(sql);
        for (int i = 0; i < 1000; i++) {
            statement.setString(1, "name_" + i);
            // 1. "攒" sql
            statement.addBatch();
            if (i % 500 == 0) {
                // 2. 执行
                statement.executeBatch();
                // 3. 清空
                statement.clearBatch();
            }
        }
        // 2. 提交数据
        connection.commit();
        JDBCUtils.closeResource(connection, statement);
        long end = System.currentTimeMillis();
        System.out.println(end - start);    // 1000 -> 1338ms
    }
}
推荐使用方式四。

 
                
            
         
         浙公网安备 33010602011771号
浙公网安备 33010602011771号