我在我准备好的语句中使用了MERGE
命令,当我在单线程环境中执行它时,它工作正常,但在多线程环境中,它会导致一些问题。这就是数据重复,也就是说,如果我有5个线程,每个记录将重复5次。我认为db中没有锁来帮助该线程。
我的代码:
//db:oracle
sb.append("MERGE INTO EMP_BONUS EB USING (SELECT 1 FROM DUAL) on (EB.EMP_id = ?) WHEN MATCHED THEN UPDATE SET TA =?,DA=?,TOTAL=?,MOTH=? WHEN NOT MATCHED THEN "+ "INSERT (EMP_ID, TA, DA, TOTAL, MOTH, NAME)VALUES(?,?,?,?,?,?) ");
//sql operation,calling from run() method
public void executeMerge(String threadName) throws Exception {
ConnectionPro cPro = new ConnectionPro();
Connection connE = cPro.getConection();
connE.setAutoCommit(false);
System.out.println(sb.toString());
System.out.println("Threadname="+threadName);
PreparedStatement pStmt= connE.prepareStatement(sb.toString());
try {
count = count + 1;
for (Employee employeeObj : employee) {//datalist of employee
pStmt.setInt(1, employeeObj.getEmp_id());
pStmt.setDouble(2, employeeObj.getSalary() * .10);
pStmt.setDouble(3, employeeObj.getSalary() * .05);
pStmt.setDouble(4, employeeObj.getSalary()
+ (employeeObj.getSalary() * .05)
+ (employeeObj.getSalary() * .10));
pStmt.setInt(5, count);
pStmt.setDouble(6, employeeObj.getEmp_id());
pStmt.setDouble(7, employeeObj.getSalary() * .10);
pStmt.setDouble(8, employeeObj.getSalary() * .05);
pStmt.setDouble(9, employeeObj.getSalary()
+ (employeeObj.getSalary() * .05)
+ (employeeObj.getSalary() * .10));
pStmt.setInt(10, count);
pStmt.setString(11, threadName);
// pStmt.executeUpdate();
pStmt.addBatch();
}
pStmt.executeBatch();
connE.commit();
} catch (Exception e) {
connE.rollback();
throw e;
} finally {
pStmt.close();
connE.close();
}
}
如果employee.size = 5,线程数= 5,执行后我会得到25条记录而不是5条
答案 0 :(得分:0)
如果没有约束(即emp_id
中emp_bonus
列上的主键或唯一键约束),则无法阻止数据库允许每个线程插入5行。由于每个数据库会话都看不到其他会话所做的未提交的更改,因此每个线程都会看到emp_bonus
中没有行与线程正在查找的emp_id
(我假设{{1} }返回每个线程中相同的5 employeeObj.getEmp_id()
个值,因此如果有5个线程,每个线程将插入所有5行,总共有25行。如果您有一个唯一的约束来阻止插入重复的行,Oracle将允许其他4个线程阻塞,直到第一个线程提交允许后续线程进行更新而不是插入。当然,这会导致线程被序列化,从而破坏运行多线程所带来的任何性能提升。