zoukankan      html  css  js  c++  java
  • JDBC的PreparedStatement启动事务使用批处理executeBatch()

    JDBC使用MySQL处理大数据的时候,自然而然的想到要使用批处理,

    普通的执行过程是:每处理一条数据,就访问一次数据库;

    而批处理是:累积到一定数量,再一次性提交到数据库,减少了与数据库的交互次数,所以效率会大大提高

    至于事务:事务指逻辑上的一组操作,组成这组操作的各个单元,要不全部成功,要不全部不成功,默认是关闭事务的。

                  更多事务的资料,请参考这里:http://blog.csdn.net/caomiao2006/article/details/22412755

    1. PreparedStatement使用批处理 executeBatch()

           1.1. 不使用executeBatch(),而使用executeUpdate()

              代码如下:  

             Class.forName("com.mysql.jdbc.Driver");
             Connection conn = DriverManager.getConnection(dbUrl, user, password);
             PreparedStatement pstmt = conn.prepareStatement("update content set introtext=? where id=?");

             for(int i=0; i<10000; i++){
                 pstmt.setString(1, "abc"+i);
                 pstmt.setInt(2, id);
                 pstmt.executeUpdate();
             }

             这样,更新10000条数据,就得访问数据库10000次

          1.2. 而使用executeBatch()

              代码如下:

             Class.forName("com.mysql.jdbc.Driver");
             Connection conn = DriverManager.getConnection(dbUrl, user, password);
             PreparedStatement pstmt = conn.prepareStatement("update content set introtext=? where id=?");

             for(int i=0; i<10000; i++){
                 pstmt.setString(1, "abc"+i);
                 pstmt.setInt(2, id);
                 pstmt.addBatch();//添加到同一个批处理中
             }

             pstmt.executeBatch();//执行批处理


             注意:1. 如果使用了 addBatch() -> executeBatch() 还是很慢,那就得使用到这个参数了

                          rewriteBatchedStatements=true (启动批处理操作)

                          在数据库连接URL后面加上这个参数:      

                              String dbUrl =  "jdbc:mysql://localhost:3306/User? rewriteBatchedStatements=true";

                          2. 在代码中,pstmt的位置不能乱放,

                              //必须放在循环体外

                         pstmt = conn.prepareStatement("update content set introtext=? where id=?");

                         for(int i=0; i<10000; i++){

                               //放这里,批处理会执行不了,因为每次循环重新生成了pstmt,不是同一个了

                               //pstmt = conn.prepareStatement("update content set introtext=? where id=?");
                               pstmt.setString(1, "abc"+i);
                               pstmt.setInt(2, id);
                               pstmt.addBatch();//添加到同一个批处理中
                         }

                         pstmt.executeBatch();//执行批处理

    2. 启用事务处理

              Class.forName("com.mysql.jdbc.Driver");

              Connection conn = DriverManager.getConnection(dbUrl, user, password);

              conn.setAutoCommit(false);//将自动提交关闭
              PreparedStatement pstmt = conn.prepareStatement("update content set introtext=? where id=?");
              pstmt.setString(1, tempintrotext);
              pstmt.setInt(2, id);
              pstmt.addBatch();
              pstmt.executeBatch();
              pstmt.close();


              conn.commit();//执行完后,手动提交事务
              conn.setAutoCommit(true);//再把自动提交打开,避免影响其他需要自动提交的操作
              conn.close();

    3. 事务和批处理混合使用      

              Class.forName("com.mysql.jdbc.Driver");

              Connection conn = DriverManager.getConnection(dbUrl, user, password);

              conn.setAutoCommit(false);//将自动提交关闭
              PreparedStatement pstmt = conn.prepareStatement("update content set introtext=? where id=?");

              for(int i=0; i<1000000; i++){
                   pstmt.setString(1, tempintrotext);
                   pstmt.setInt(2, id);
                   pstmt.addBatch();

                   //每500条执行一次,避免内存不够的情况,可参考,Eclipse设置JVM的内存参数

                   if(i>0 && i%500==0){
                        pstmt.executeBatch();

                        //如果不想出错后,完全没保留数据,则可以每执行一次提交一次,但得保证数据不会重复

                        conn.commit();

                    }

             }
              pstmt.executeBatch();//执行最后剩下不够500条的
              pstmt.close();

              conn.commit();//执行完后,手动提交事务
              conn.setAutoCommit(true);//再把自动提交打开,避免影响其他需要自动提交的操作
              conn.close();

     较完整的代码:

     1 public class ExecuteBatchTest {
     2     private Connection conn;
     3     private PreparedStatement pstmt;
     4     private PreparedStatement pstmt2;
     5     private ResultSet rs;
     6     private String user = "root";
     7     private String password = "123456";
     8     private String dbUrl = "jdbc:mysql://localhost:3306/user?rewriteBatchedStatements=true";
     9     private int limitNum = 10000;
    10 
    11     public void changeData() {
    12         try {
    13             Class.forName("com.mysql.jdbc.Driver");
    14             conn = DriverManager.getConnection(dbUrl, user, password);
    15             
    16             //既不用batch,也不用事务
    17             testBatch(false,false);
    18             //只用batch, 不用事务
    19             testBatch(false,true);
    20             //只用事务,不用batch
    21             testBatch(true,false);
    22             //不仅用事务,还用batch
    23             testBatch(true,true);
    24             
    25             pstmt.close();
    26             conn.close();
    27         } catch (ClassNotFoundException e) {
    28             e.printStackTrace();
    29         } catch (SQLException e) {
    30             e.printStackTrace();
    31         }
    32     }
    33     
    34     public void testBatch(Boolean openTransaction, Boolean useBatch) throws SQLException{
    35         if(openTransaction)
    36             conn.setAutoCommit(false);
    37         
    38         if(pstmt!=null){
    39             pstmt.clearParameters();
    40             pstmt.clearBatch();
    41         }
    42         
    43         pstmt = conn.prepareStatement("insert into person (name) values (?)");
    44         long start = System.currentTimeMillis();
    45         for(int a = 0;a<limitNum;a++){
    46             String name = "tommy"+a;
    47             pstmt.setString(1, name);
    48             if(useBatch)
    49                 pstmt.addBatch();
    50             else
    51                 pstmt.executeUpdate();
    52         }
    53         
    54         if(useBatch)
    55            pstmt.executeBatch();
    56          
    57         if(openTransaction){
    58             conn.commit();
    59             conn.setAutoCommit(true);
    60         }
    61         long end = System.currentTimeMillis();
    62         System.out.println("use time:"+(end-start)+" ms");
    63         
    64     }
    65     
    66     //main method
    67     publi static void main(String[] args){
    68         ExecuteBatchTest ebt = new ExecuteBatchTest();
    69         ebt.changeData();
    70     }
    71        
    72 }
    View Code

    运行结果:

       分别是: 不用批处理,不用事务;

                   只用批处理,不用事务;

                   只用事务,不用批处理;

                   既用事务,也用批处理;(很明显,这个最快,所以建议在处理大批量的数据时,同时使用批处理和事务)

     
  • 相关阅读:
    easy-animation | Animation for Sass
    UC手机浏览器(U3内核)相关文档整理
    视差滚动(Parallax Scrolling)的一点小心得
    MVC Filter 返回json格式
    Jquery 实现左右两侧菜单添加、移除
    .Net 发送邮件
    Request.From,Request.QueryString转对象
    DataTable转List,转对象
    SQL Table 自动生成Net底层-控制器Autofac注入
    SQL Table 自动生成Net底层-生成业务层Service
  • 原文地址:https://www.cnblogs.com/tommy-huang/p/4540407.html
Copyright © 2011-2022 走看看