对于对象的同步和异步的方法,我们在设计自己的程序的时候,一定要考虑问题的整体,不然就会出现数据不一致的情况,很经典的错误就是脏读。
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:数据库事务正确执行四要素,原子性、一致性、隔离性、持久性