每天一例多线程[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

避免了脏读数据的产生。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

jeff.sheng

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值