每天一例多线程[day4]-----synchronized与DirtyRead脏读问题

问题:以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

避免了脏读数据的产生。

猜你喜欢

转载自blog.csdn.net/shengqianfeng/article/details/80557180