首頁 >Java >java教程 >Java如何實作JDBC批次插入

Java如何實作JDBC批次插入

PHPz
PHPz轉載
2023-05-18 10:02:021955瀏覽

    一、說明

    在JDBC中,executeBatch這個方法可以將多條dml語句批次執行,效率比單一執行executeUpdate高很多,這是什麼原理呢?在mysql和oracle中又是如何實現批次執行的呢?本文將為大家介紹背後的原理。

    二、實驗介紹

    本實驗將透過以下三步驟進行

    a. 記錄jdbc在mysql中批次執行和單一執行的耗時

    b. 記錄jdbc在oracle中批次執行和單一執行的耗時

    c. 記錄oracle plsql批次執行和單一執行的耗時

    相關java和資料庫版本如下:Java17, Mysql8,Oracle11G

    三、正式實驗

    在mysql和oracle中分別建立一張表格

    create table t (  -- mysql中创建表的语句
        id    int,
        name1 varchar(100),
        name2 varchar(100),
        name3 varchar(100),
        name4 varchar(100)
    );
    create table t (  -- oracle中创建表的语句
        id    number,
        name1 varchar2(100),
        name2 varchar2(100),
        name3 varchar2(100),
        name4 varchar2(100)
    );

    在實驗前需要開啟資料庫的稽核

    mysql開啟審計:

    set global general_log = 1;

    oracle開啟審計:

    alter system set audit_trail=db, extended;  
    audit insert table by scott;  -- 实验采用scott用户批量执行insert的方式

    java程式碼如下:

    import java.sql.*;
    
    public class JdbcBatchTest {
    
        /**
         * @param dbType 数据库类型,oracle或mysql
         * @param totalCnt 插入的总行数
         * @param batchCnt 每批次插入的行数,0表示单条插入
         */
        public static void exec(String dbType, int totalCnt, int batchCnt) throws SQLException, ClassNotFoundException {
            String user = "scott";
            String password = "xxxx";
            String driver;
            String url;
            if (dbType.equals("mysql")) {
                driver = "com.mysql.cj.jdbc.Driver";
                url = "jdbc:mysql://ip/hello?useServerPrepStmts=true&rewriteBatchedStatements=true";
            } else {
                driver = "oracle.jdbc.OracleDriver";
                url = "jdbc:oracle:thin:@ip:orcl";
            }
    
            long l1 = System.currentTimeMillis();
            Class.forName(driver);
            Connection connection = DriverManager.getConnection(url, user, password);
            connection.setAutoCommit(false);
            String sql = "insert into t values (?, ?, ?, ?, ?)";
            PreparedStatement preparedStatement = connection.prepareStatement(sql);
            for (int i = 1; i <= totalCnt; i++) {
                preparedStatement.setInt(1, i);
                preparedStatement.setString(2, "red" + i);
                preparedStatement.setString(3, "yel" + i);
                preparedStatement.setString(4, "bal" + i);
                preparedStatement.setString(5, "pin" + i);
    
                if (batchCnt > 0) {
                    // 批量执行
                    preparedStatement.addBatch();
                    if (i % batchCnt == 0) {
                        preparedStatement.executeBatch();
                    } else if (i == totalCnt) {
                        preparedStatement.executeBatch();
                    }
                } else {
                    // 单条执行
                    preparedStatement.executeUpdate();
                }
            }
            connection.commit();
            connection.close();
            long l2 = System.currentTimeMillis();
            System.out.println("总条数:" + totalCnt + (batchCnt>0? (",每批插入:"+batchCnt) : ",单条插入") + ",一共耗时:"+ (l2-l1) + " 毫秒");
        }
    
        public static void main(String[] args) throws SQLException, ClassNotFoundException {
            exec("mysql", 10000, 50);
        }
    }

    程式碼中幾個注意的點,

    • #mysql的url需要加入useServerPrepStmts=true&rewriteBatchedStatements=true參數。

    • batchCnt表示每次批次執行的sql條數,0表示單一執行。

    先測試mysql

    exec("mysql", 10000, batchCnt);

    代入不同的batchCnt值看執行時長

    batchCnt=50 總條數:10000,每批插入:50,共耗時:4369 毫秒
    batchCnt=100 總條數:10000,每批插入:100,共耗時:2598 毫秒
    batchCnt=200 總條數:10000,每批插入: 200,共耗時:2211 毫秒
    batchCnt=1000 總條數:10000,每批插入:1000,共耗時:2099 毫秒
    batchCnt=10000 總條數:10000,每批插入:10000共耗時:2418 毫秒
    batchCnt=0 總條數:10000,單一插入,共耗時:59620 毫秒

    ##查看general log

    batchCnt=5

    batchCnt=0

    可以得到幾個結論:

    • #批次執行的效率比起單條執行大幅提升。

    • mysql的批次執行其實是改寫了sql,將多條insert合併成了insert xx values(),()...的方式去執行。

    • 將batchCnt由50改到100的時候,時間基本上縮短了一半,但是再擴大這個值的時候,時間縮短並不明顯,執行的時間甚至還會升高。

    分析原因:

    客戶端將要執行的SQL語句傳送給資料庫伺服器後,資料庫執行該SQL語句並將結果傳回給客戶端。總耗時 = 資料庫執行時間 網路傳輸時間。透過大量執行減少往返次數可以降低網路傳輸時間,從而縮短總時間。然而,當batchCnt變大時,即使網路傳輸時間不再是最主要的瓶頸,總時間的降低也不會那麼明顯。特別是當batchCnt=10000,即一次性把1萬條語句全部執行完,時間反而變多了,這可能是由於程式和資料庫在準備這些入參時需要申請更大的內存,所以耗時更多(我猜的)。

    再來說一句,batchCnt這個值是不是能無限大呢,假設我需要插入的是1億條,那麼我能一次批量插入1億條嗎?當然不行,我們不考慮undo的空間問題,首先你電腦就沒有這麼大的內存一次性把這1億條sql的入參全部保存下來,其次mysql還有個參數max_allowed_pa​​cket限制單條語句的長度,最大為1G位元組。當語句過長的時候就會報"Packet for query is too large (1,773,901 > 1,599,488). You can change this value on the server by setting the 'max_allowed_pa​​cket' variable"。

    接下來測試oracle

    exec("oracle", 10000, batchCnt);

    代入不同的batchCnt值看執行時長

    batchCnt=50 總條數:10000,每批插入:50,一共耗時:2055 毫秒

    batchCnt=100 總條數:10000,每批插入:100,共耗時:1324 毫秒
    batchCnt=200 總條數:10000,每批插入:200,總共耗時:856 毫秒
    batchCnt=1000 總條數:10000,每批插入:1000,總共耗時:785 毫秒
    batchCnt=10000 總條數:10000,每批插入:10000,總共耗時:804毫秒
    batchCnt=0 總條數:10000,單一插入,總共耗時:60830 毫秒

    在Oracle中執行的效果跟MySQL中基本上一致,批次操作的效率明顯高於單條執行。問題就來了,oracle中並沒有這種insert xx values(),()..語法呀,那它是怎麼做到批量執行的呢?

    查看當執行batchCnt=50的稽核檢視dba_audit_trail

    从审计的结果中可以看到,batchCnt=50的时候,审计记录只有200条(扣除登入和登出),也就是sql只执行了200次。sql_text没有发生改写,仍然是"insert into t values (:1 , :2 , :3 , :4 , :5 )",而且sql_bind只记录了批量执行的最后一个参数,即50的倍数。根据awr报告可以看出,实际只执行了200次(由于篇幅限制,省略了awr截图)。那么oracle是怎么做到只执行200次但插入1万条记录的呢?我们来看看oracle中使用存储过程的批量插入。

    四、存储过程

    准备数据:

    首先将t表清空 truncate table t;

    用java往t表灌10万数据 exec("oracle", 100000, 1000);

    创建t1表 create table t1 as select * from t where 1 = 0;

    以下两个过程的意图一致,均为将t表中的数据导入t1表。nobatch是单次执行,usebatch是批量执行。

    create or replace procedure nobatch is
    begin
      for x in (select * from t)
      loop
        insert into t1 (id, name1, name2, name3, name4)
        values (x.id, x.name1, x.name2, x.name3, x.name4);
      end loop;
      commit;
    end nobatch;
    /
    create or replace procedure usebatch (p_array_size in pls_integer)
    is
      type array is table of t%rowtype;
      l_data array;
      cursor c is select * from t;
    begin
      open c;
      loop
        fetch c bulk collect into l_data limit p_array_size;
        forall i in 1..l_data.count insert into t1 values l_data(i);
        exit when c%notfound;
      end loop;
      commit;
      close c;
    end usebatch;
    /

    执行上述存储过程

    SQL> exec nobatch;  
    Elapsed: 00:00:32.92

    SQL> exec usebatch(50);
    Elapsed: 00:00:00.77

    SQL> exec usebatch(100);
    Elapsed: 00:00:00.47

    SQL> exec usebatch(1000);
    Elapsed: 00:00:00.19

    SQL> exec usebatch(100000);
    Elapsed: 00:00:00.26

    存储过程批量执行效率也远远高于单条执行。查看usebatch(50)执行时的审计日志,sql_bind也只记录了批量执行的最后一个参数,即50的倍数。与使用executeBatch方法在记录内容方面相同。因此可以推断,JDBC的executeBatch和存储过程的批量执行都采用了相同的方法

    存储过程的这个关键点就是forall。查阅相关文档。

    The FORALL statement runs one DML statement multiple times, with different values in the VALUES and WHERE clauses.
    The different values come from existing, populated collections or host arrays. The FORALL statement is usually much faster than an equivalent FOR LOOP statement.
    The FORALL syntax allows us to bind the contents of a collection to a single DML statement, allowing the DML to be run for each row in the collection without requiring a context switch each time.

    翻译过来就是forall很快,原因就是不需要每次执行的时候等待参数。

    以上是Java如何實作JDBC批次插入的詳細內容。更多資訊請關注PHP中文網其他相關文章!

    陳述:
    本文轉載於:yisu.com。如有侵權,請聯絡admin@php.cn刪除