一次同步方法的失败与原因总结
2018-12-27 07:42:25来源:博客园 阅读 ()
synchronized关键字可以设置同步方法和同步块,同步方法会对调用对象this上锁,
所有线程进入前都需要获取这个锁,这里我拿一个错误样例来做示范
1 public class Test { 2 public static void main(String[] args) { 3 Account account=new Account(100); 4 Person p1=new Person(account,80); 5 Person p2=new Person(account,90); 6 p1.start(); 7 p2.start(); 8 } 9 } 10 11 class Account{ 12 int total; 13 public Account(int total) { 14 super(); 15 this.total=total; 16 } 17 } 18 19 class Person extends Thread{ 20 private int reduce; 21 private Account account; 22 public Person(Account account,int reduce) { 23 24 this.account=account; 25 this.reduce=reduce; 26 } 27 28 public synchronized void run() { 29 if (account.total-reduce<0) return ; 30 try { 31 Thread.sleep(200); 32 } catch (InterruptedException e) { 33 // TODO Auto-generated catch block 34 e.printStackTrace(); 35 } 36 account.total-=reduce; 37 System.out.println(Thread.currentThread().getName()+"取出"+reduce); 38 System.out.println(Thread.currentThread().getName()+"剩余"+account.total); 39 40 } 41 }
Thread-0取出80
Thread-0剩余-70
Thread-1取出90
Thread-1剩余-70
出现了负数,很明显没锁住,现在我们来分析下原因。
有1个account对象,两个person对象p1和p2,p1和p2争抢资源account
我一开始的设想是,当p1进入方法时,p1对象上锁,account作为p1的成员也被上锁,此时p2进入就需要等到p1释放锁,由此达到我们的目标。
问题在于:java规定每个对象都有一个监视器,使用时查看上锁了没,上面的代码中虽然对person对象上了锁,同时account作为对象也有监视器,account虽然作为person的成员,但account的监视器是单独的,不受p1的锁影响。现在我们再来场景复原一下,p1先进入方法,对p1上锁,计算100-80=20,同时p2进入方法,查看account的监视器,没有上锁,计算20-90=-70,这就产生了错误。
解决方法就是使用同步块给account上锁
1 public void run() { 2 synchronized(account) { 3 if (account.total-reduce<0) return ; 4 try { 5 Thread.sleep(200); 6 } catch (InterruptedException e) { 7 // TODO Auto-generated catch block 8 e.printStackTrace(); 9 } 10 account.total-=reduce; 11 System.out.println(Thread.currentThread().getName()+"取出"+reduce); 12 System.out.println(Thread.currentThread().getName()+"剩余"+account.total); 13 14 } 15 }
这个问题给我们的教训在于,每个对象都有单独的监视器,要选取正确的争夺对象上锁,基于此同步块不仅能缩小锁的粒度还能选取正确的对象上锁。
标签:
版权申明:本站文章部分自网络,如有侵权,请联系:west999com@outlook.com
特别注意:本站所有转载文章言论不代表本站观点,本站所提供的摄影照片,插画,设计作品,如需使用,请与原作者联系,版权归原作者所有
下一篇:年终总结
- 类的继承,方法重新中修饰符如何定义 2020-06-10
- java里面main方法中的String[]args 2020-06-07
- 错误: 在类中找不到 main 方法, 请将 main 方法定义为: & 2020-06-06
- 学习笔记之方法引用 2020-06-06
- Java连载120-反射机制获取构造方法和父类、父接口 2020-06-05
IDC资讯: 主机资讯 注册资讯 托管资讯 vps资讯 网站建设
网站运营: 建站经验 策划盈利 搜索优化 网站推广 免费资源
网络编程: Asp.Net编程 Asp编程 Php编程 Xml编程 Access Mssql Mysql 其它
服务器技术: Web服务器 Ftp服务器 Mail服务器 Dns服务器 安全防护
软件技巧: 其它软件 Word Excel Powerpoint Ghost Vista QQ空间 QQ FlashGet 迅雷
网页制作: FrontPages Dreamweaver Javascript css photoshop fireworks Flash