1.批处理
在我们进行大批量数据操作的时候,需要采用批处理的方式来提高程序的运行性能,目的是减少跟数据库交互的次数
注意:executeBatch()方法返回值为int数组
案例
①.Statement实现批处理
@Test
public void batchTest1() {
Connection conn = DBUtils.getConnection();
Statement st = null;
try {
st = conn.createStatement();
st.addBatch("insert into t_user values(2,'xx','123')");
st.addBatch("insert into t_user values(3,'cc','123')");
st.executeBatch();
} catch (SQLException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}finally {
CloseUtils.closeAll(st,conn);
}
}
②.PreparedStatement实现批处理
@Test
public void batchTest2() {
Connection conn = DBUtils.getConnection();
PreparedStatement prst = null;
try {
String sql = "insert into t_user(name,pw) values(?,?)";
prst = conn.prepareStatement(sql);
for(int i=0;i<1000;i++){
prst.setString(1, "zz"+i);
prst.setInt(2, 123+i);
prst.addBatch();
if(i%100==0)
prst.executeBatch();
}
prst.executeBatch();
} catch (SQLException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}finally {
CloseUtils.closeAll(prst,conn);
}
}
2.事务
MySQL:每一条语句都属于独立事务,默认自动管理提交的。
如果需要把多条语句当成一个整体,那么就需要把多条语句放在一个事务里面
开启事务:start transaction
提交事务:commit;
回滚事务:rollback
JDBC控制事务语句
Connection.setAutoCommit(false); //start transaction
Connection.rollback(); //rollback
Connection.commit(); //commit
模拟一个转账的功能,用事务来控制
案例:转账功能–使用事务
import java.sql.Connection;
import java.sql.SQLException;
import java.sql.Statement;
import com.qf.utils.CloseUtils;
import com.qf.utils.DBUtils;
//案例:转账功能--使用事务
public class CommitTest {
public static void main(String[] args) {
Connection conn = null;
Statement st = null;
try {
conn = DBUtils.getConnection();
conn.setAutoCommit(false);// 关闭自动提交(开启事务) 默认为true
st = conn.createStatement();
int result = st.executeUpdate("update t_bank set balance=balance-2000 where name='zs'");
System.out.println(result);
// 出现异常
// int i = 1/0;
result = st.executeUpdate("update t_bank set balance=balance+2000 where name='ls'");
System.out.println(result);
conn.commit();
} catch (Exception e) {
System.out.println("回滚");
try {
conn.rollback();
} catch (SQLException e1) {
// TODO Auto-generated catch block
e1.printStackTrace();
} // 回滚
} finally {
CloseUtils.closeAll(st, conn);
}
}
}
事务的特点
事务的特性:ACID
原子性( Atomicity )、一致性( Consistency )、隔离性( Isolation )和持久性( Durability )
原子性:事务是数据库的逻辑工作单位,事务中包含的各操作要么都完成,要么都不完成
一致性:事务执行的结果必须是使数据库从一个一致性状态变到另一个一致性状态。因此当数据库只包含成功事务提交的结果时,就说数据库处于一致性状态。如果数据库系统 运行中发生故障,有些事务尚未完成就被迫中断,这些未完成事务对数据库所做的修改有一部分已写入物理数据库,这时数据库就处于一种不正确的状态,或者说是 不一致的状态。
隔离性:一个事务的执行不能其它事务干扰。即一个事务内部的操作及使用的数据对其它并发事务是隔离的,并发执行的各个事务之间不能互相干扰。
持久性:指一个事务一旦提交,它对数据库中的数据的改变就应该是永久性的。接下来的其它操作或故障不应该对其执行结果有任何影响。
事务的隔离级别:
属于事务的。都已开启了事务为前提。
不考虑事务的隔离级别,会出现以下错误的情况
- 脏读:一个线程中的事务读到了另外一个线程中未提交的数据。
- 不可重复读:一个线程中的事务读到了另外一个线程中已经提交的update的数据。
- 虚读:一个线程中的事务读到了另外一个线程中已经提交的insert的数据。
要想避免以上现象,通过更改事务的隔离级别来避免:
- READ UNCOMMITTED 脏读、不可重复读、虚读有可能发生。
- READ COMMITTED 避免脏读的发生,不可重复读、虚读有可能发生。
- REPEATABLE READ 避免脏读、不可重复读的发生,虚读有可能发生。
- SERIALIZABLE 避免脏读、不可重复读、虚读的发生。
级别依次升高,效率依次降低。
MySQL:默认REPEATABLE READ
ORACLE:默认READ COMMITTED
MySQL:
select @@tx_isolation;//查看当前的隔离级别
set transaction isolation level 级别;// 设置当前的事务隔离级别
练习:read uncommitted;
开启两个cmd窗口即可进行测试
JDBC设置隔离级别
Connection,在开启事务之前进行设置