- 1、本文档共53页,可阅读全部内容。
- 2、有哪些信誉好的足球投注网站(book118)网站文档一经付费(服务费),不意味着购买了该文档的版权,仅供个人/单位学习、研究之用,不得用于商业用途,未经授权,严禁复制、发行、汇编、翻译或者网络传播等,侵权必究。
- 3、本站所有内容均由合作方或网友上传,本站不对文档的完整性、权威性及其观点立场正确性做任何保证或承诺!文档内容仅供研究参考,付费前请自行鉴别。如您付费,意味着您自己接受本站规则且自行承担风险,本站不退款、不进行额外附加服务;查看《如何避免下载的几个坑》。如果您已付费下载过本站文档,您可以点击 这里二次下载。
- 4、如文档侵犯商业秘密、侵犯著作权、侵犯人身权等,请点击“版权申诉”(推荐),也可以打举报电话:400-050-0827(电话支持时间:9:00-18:30)。
查看更多
ch10_线程.ppt
多线程并发执行中的问题 多个线程相对执行的顺序是不确定的。 线程执行顺序的不确定性会产生执行结果的不确定性。 在多线程对共享数据 操作时常常会产生这种不确定性。 多线程并发执行中的问题 一个堆栈类: public class MyStack{ private int idx = 0 ; private char[ ] data = new char[6] ; public void push( char c ){ data[idx] = c ; idx ++ ; // 栈顶指针指向下一个空单元 } public char pop( ){ idx-- ; return data[idx] ; } } 多线程并发执行中的问题 线程 a 线程b 堆栈 s 状态 时刻t1 时刻t2 时刻t3 //调用s.Push( ); data[idx]=‘r’ ; //调用s.pop( ) idx-- ; return data[idx]; //恢复运行 idx++ ; /*线程a 被抢占 idx++; 未执行*/ | p | q | | | | | idx=2 | p | q | r | | | | idx=2 | p | q | r | | | | idx=1 | p | q | r | | | | idx=2 . . . !“q”被两次压栈; 线程a 压入的数据“r”丢失。 对象锁的概念 Java中每个对象都带有一个monitor标志,相当于一个锁。 Sychronized关键字用来给对象加上独占的排它锁。 对象锁的操作 Code or behavior Data or state Object this Thread before synchronized(this) public void push(char c){ synchronized(this){ data[idx] = c ; idx++ ; } } 对象锁的操作 Code or behavior Data or state Object this Thread after synchronized(this) public void push(char c){ synchronized(this){ data[idx] = c ; idx++ ; } } 示例 将MyStack 对栈的push,pop操作都采用这种机制: public class MyStack{ … public void push( char c ){ synchronized(this){ data[idx] = c ; idx ++ ; } } public char pop( ){ synchronized(this){ idx-- ; return data[idx] ;} } } 多线程对共享数据进行操作 线程 a 线程b 堆栈 s 状态 时刻t1 时刻t2 时刻t3 /*调 用s.push( ), 获得s的monitor后运行*/ data[idx]=‘r’ ; /*调用s.pop(),未获得s的monitor, b到s的lock pool中等待*/ //恢复运行 idx++ ; 。。。 /*完成,并交回S的 monitor*/ //线程a 被抢占 | p | q | r | | | | idx=2 | p | q | r | | | | idx=2 | p | q | r | | | | idx=3 . . . 时刻t4 //运行pop … 得结果r | p | q | r | | | | idx=2 几点说明 如何返还对象的monitor 当synchronized( ){ }语句块执行完毕后。 当在synchronized( ){ }语句块中出现exception. 当调用该对象的wait( )方法。将该线程放入对象的wait pool中,等待某事件的发生。 几点说明 对共享数据的所有访问都必须使用synchronized. 用synchronized保护的共享数据必须是私有的,使线程 不能直接访问这些数据,必须通过对象的方法。 如果一个方法的整体都在synchronized块中,则可以把synchronized关键字放于方法定义的头部: public synchronized void push( char c){ … } 几点说明
文档评论(0)