脏读一

对于对象的同步和异步的方法,我们在设计自己的程序的时候,一定要考虑问题的整体,不然就会出现数据不一致的情况,很经典的错误就是脏读。

package com.ljq.test;

/**
 * 业务整体需要使用完整的synchronized,保持业务的原子性。
 * 
 */
public class DirtyRead {
    private String username = "bjsxt";
    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);
    }

    public synchronized 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();
    }

}

a、getValue方法去掉synchronized关键字,控制台打印信息:
getValue方法得到:username = z3 , password = 123
setValue最终结果:username = z3 , password = 456

b、getValue方法加上synchronized关键字,控制台打印信息:
setValue最终结果:username = z3 , password = 456
getValue方法得到:username = z3 , password = 456

示例总结:
在我们对一个对象的方法加锁的时候,需要考虑业务的整体性,即为setValue/getValue方法同时加锁synchronized同步关键字,保证业务(service)的原子性,不然会出现业务错误(也从侧面保证业务的一致性)。

acid:数据库事务正确执行四要素,原子性、一致性、隔离性、持久性

原文地址:https://www.cnblogs.com/linjiqin/p/5839034.html