探索并发编程..docx

  1. 1、本文档共28页,可阅读全部内容。
  2. 2、有哪些信誉好的足球投注网站(book118)网站文档一经付费(服务费),不意味着购买了该文档的版权,仅供个人/单位学习、研究之用,不得用于商业用途,未经授权,严禁复制、发行、汇编、翻译或者网络传播等,侵权必究。
  3. 3、本站所有内容均由合作方或网友上传,本站不对文档的完整性、权威性及其观点立场正确性做任何保证或承诺!文档内容仅供研究参考,付费前请自行鉴别。如您付费,意味着您自己接受本站规则且自行承担风险,本站不退款、不进行额外附加服务;查看《如何避免下载的几个坑》。如果您已付费下载过本站文档,您可以点击 这里二次下载
  4. 4、如文档侵犯商业秘密、侵犯著作权、侵犯人身权等,请点击“版权申诉”(推荐),也可以打举报电话:400-050-0827(电话支持时间:9:00-18:30)。
查看更多
探索并发编程.

/cutesource/article/details/5778820探索并发编程(一)操作系统篇在多线程、多处理器甚至是分布式环境的编程时代,并发是一个不可回避的问题,很多程序员一碰到并发二字头皮就发麻,也包括我。既然并发问题摆在面前一个到无法回避的坎,倒不如拥抱它,把它搞清楚,决心花一定的时间从操作系统底层原理到Java的基础编程再到分布式环境等几个方面深入探索并发问题。先就从原理开始吧。并发产生的原因虽然从直观效果上,处理器是并行处理多项任务,但本质上一个处理器在某个时间点只能处理一个任务,属于串行执行。在单处理器的情况下,并发问题源于多道程序设计系统的一个基本特性:进程的相对执行速度不可预测,它取决于其他进程的活动、操作系统处理中断的方式以及操作系统的调度策略。在分布式环境下,并发产生的可能性就更大了,只要大家有依赖的共享资源,就有并发问题的出现,因为互相调用次序更加没法控制。并发带来的问题全局资源的共享充满了危险。不同任务对同一个共享资源的读写顺序非常关键操作系统很难对分配资源进行最优化管理。挂起的线程占有了其他活动线程需要的资源定位错误非常困难。这种问题来源和触发的不确定性,导致定位问题非常困难限制分布式系统横向扩展能力进程的交互进程的交互方式决定了并发问题产生的上下文,解决并发问题也需根据进程交互方式的不同而不同对待。一般进程交互分为以下三种:1)进程间相互独立这种情况下虽然进程间没有数据共享,所做事情也互不联系,但它们存在竞争关系。计算机中有些临界资源比如I/O设备、存储器、CPU时间和时钟等等都需要通过竞争得到,你占用的时候就得保证别人没法占用,因此首先得解决这种互斥的需求。另外,要处理好这种临界资源的调度策略,处理不当就有可能发生死锁和饥饿2)进程间通过共享合作这种情况下进程间虽然执行的过程是相互独立的,互不知道对方的执行情况,但互相之间有共享的数据。因此除了有以上互斥需求和死锁饥饿的可能,另外还会有数据一致性的问题。当多个进程非原子性操作同一个数据时候,互相之间操作时序不当就有可能造成数据不一致3)进程间通过通信合作这种情况下进程间通过消息互相通信,知晓各自的执行情况,不共享任何资源,因此就可以避免互斥和数据不一致问题,但仍然存在死锁和饥饿的问题并发问题的解决办法操作系统解决并发问题一般通过互斥,为了提供互斥的支持,需要满足以下需求:一次只允许一个进程进入临界区一个非临界区停止的进程必须不干涉其他进程不允许出现一个需要访问临界区的进程被无限延迟一个进程驻留在临界区中的时间必须是有限的临界区空闲时,任何需要进入临界区的进程必须能够立即进入满足互斥的解决方案:1)硬件支持中断禁用中断禁用简单说来就是在某一进程在临界区执行过程中禁用中断,不允许其他进程通过中断打断其执行。虽然这种方式可以保证互斥,但代价非常高,处理器被限制于只能交替执行程序,效率降低。另外不适用于多处理器环境。专用机器指令从硬件的角度提供一些机器指令,用于保证多个动作的原子性,通过适用这些具有原子性的指令来控制临界区的访问。比如提供符合以下逻辑的原子性指令:boolean?testset(int?i){???if(i==0){???i=1;???return?true;???}else{???return?false;???}??}??在控制临界区的时候可以通过忙等待来保证只有一个进程停留在临界区,伪代码如下所示:int?bolt;??void?onlyOneThread(){???while(!testset(bolt)){???/*等待*/??}???/*临界区*/??bolt=0;??}??专用机器指令的优点是可以不限制处理器数量,也不限制临界区的数量,但它的问题是使用了忙等待,消耗处理器时间。并且也存在饥饿和死锁的问题2)信号量其原理是多个进程可以通过简单的信号进行合作,一个进程可以被迫在某一个位置停止,直到它收到一个特定的信号,再重新被唤起工作。这种方式最大优点就是解决了忙等待的问题。其核心和机器指令类似,通过提供原子性信号量控制方法,一般情况下提供等待和唤起两种操作原语,以较为简单的二元信号量原语为例,两种方法的伪代码如下:void?wait(semaphore?s){???if(s.value==1){???s.value=0;???}else{???/*停止此线程,并把线程放入s的线程等待队列(s.queue)里*/??}??}??void?signal(semaphore?s){???if(s.queue.size()==0){???s.value=1;???}else{???/*从s的线程等待队列(s.queue)里拿出一个线程,使其激活执行*/??}??}??两个方法的实现关键在于其原子性,当然也可以借助专用机器指令的方法

文档评论(0)

stzs + 关注
实名认证
内容提供者

该用户很懒,什么也没介绍

1亿VIP精品文档

相关文档