问题:以oracle数据库为例,比如当我们在下午2点钟查询用户表的一条记录值100,由于网络卡顿,数据库直到2点半才查询出来这条记录,在这半小时内,其他用户将用户表的这条记录修改为200了,那么请问2点半我们看到的这条数据是100还是200?
答案是100,原因是oracle数据库存在一致性读的特性,不管修改的次数再多,即使抛出异常也不会看到200这个数据。
在多线程应用中,当我们在设置多项数据的时候,如果我们想在获取数据的时候得到的是设置完后的完整数据,那么我们可以用synchronized对象锁来控制。比如:
package com.jeff.base.sync004;
/**
* 业务整体需要使用完整的synchronized,保持业务的原子性。
*
* @author jeff
*
*/
public class DirtyRead {
private String username = "jeff";
private String password = "123";
public synchronized void setValue(String username, String password){
this.username = username;
try {
Thread.sleep(2000);
} catch (InterruptedException e) {
e.printStackTrace();
}
this.password = password;
System.out.println("setValue最终结果:username = " + username + " , password = " + password);
}
/**
* synchronized,可以用来保证setValue和getValue方法能够同步
* */
public void getValue(){
System.out.println("getValue方法得到:username = " + this.username + " , password = " + this.password);
}
public static void main(String[] args) throws Exception{
final DirtyRead dr = new DirtyRead();
Thread t1 = new Thread(new Runnable() {
@Override
public void run() {
dr.setValue("z3", "456");
}
});
t1.start();
Thread.sleep(1000);
dr.getValue();
}
}
打印结果:
getValue方法得到:username = z3 , password = 123
setValue最终结果:username = z3 , password = 456
分析:setValue方法没有执行完成前,getvalue就获取结果,那么最终得到的就是没设置的数据,如password是123,我们可以在getValue上加上synchronized关键词进行控制,当t1线程进行setValue设置时拿到了dr对象锁,执行完以后,主线程dr.getValue()方法才获取到被释放的对象锁,得到最终完成的结果:
setValue最终结果:username = z3 , password = 456
getValue方法得到:username = z3 , password = 456
避免了脏读数据的产生。