首页 > 代码库 > 脏读一

脏读一

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

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:数据库事务正确执行四要素,原子性、一致性、隔离性、持久性

脏读一