对于对象的同步和异步的方法,我们在设计自己程序的时候,一定要考虑问题的整体性,不然就会出现数据不一致的错误,脏读。
看代码
/**
* 业务整体需要使用完整的synchronized,保持业务的原子性。
*/
public class DirtyRead {
private String username = "helei";
private String password = "123";
public synchronized void setValue(String username, String password){
//传两个参数,对成员变量修改
this.username = username;
try {
Thread.sleep(2000);//先对用户名进行了一个设置,2秒钟以后,又对密码进行设置一次
} catch (InterruptedException e) {
e.printStackTrace();
}
this.password = password;
System.out.println("setValue最终结果:username = " + username + " , password = " + password);
}
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();//实例化一个对象,启动一个t1线程,
Thread t1 = new Thread(new Runnable() {
@Override
public void run() {
dr.setValue("helei", "456");
}
});
t1.start();
Thread.sleep(1000);
dr.getValue();
}
}
打印结果如下
那好我们分析下
主要有两个纵向线程,首先是一个主线程main,还有一个t1线程,t1执run方法,等待两秒钟
如果打印是 helei 123的话,代码中的逻辑刚执行一半,我不希望有其他线程进来,那么我们应该考虑问题的时候考虑问题的整体性, 也就是在setValue不允许getValue执行,就把两个方法上都加上synchronized
总结:在我们对一个对象的方法加锁的时候,需要考虑业务的整体性,即为setValue和getValue同时加锁synchronized同步关键字,保证业务的原子性,不然会出现业务错误。(也从侧面保证业务的一致性)
关系型数据库的四个特性
- A原子性
-
C一致性
- I隔离性
- D永久性