`
农村哥们
  • 浏览: 293256 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

转载探索并发编程(一)------操作系统篇

阅读更多
这篇总结很好,收藏共大家学习
转载http://blog.csdn.net/cutesource/archive/2010/07/31/5778820.aspx
作者:cutesource

在多线程、多处理器甚至是分布式环境的编程时代,并发是一个不可回避的问题,很多程序员一碰到并发二字头皮就发麻,也包括我。既然并发问题摆在面前一个到无法回避的坎,倒不如拥抱它,把它搞清楚,决心花一定的时间从操作系统底层原理到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)里拿出一个线程,使其激活执行*/ 
    }  
}

两个方法的实现关键在于其原子性,当然也可以借助专用机器指令的方法来保障其原子性,毕竟这两种方法的执行不长,使用忙等待也问题不大。

再看互斥的问题,若使用信号量,则其具体实现如以下伪代码所示:

void onlyOneThread(){  
    wait(s);  
    /*临界区*/ 
    signal(s);  


3)管程

信号量虽然解决了性能问题,但使得信号量的控制逻辑遍布在程序里,控制逻辑复杂以后很难整体上控制所有信号量。而管程的思路和面向对象类似,通过一个管程的概念把互斥和资源保护的细节封装在管程的内部,外部程序只需对管程的正确使用就能保证避免并发问题,管程的特点如下:

共享数据变量只能被管程的过程访问
一个进程通过调用管程的一个过程进入管程
只能有一个进程在管程中执行,其他进程被挂起,等待进入管程
4)消息传递

消息传递是通过消息通信的方式进程之间相互配合,满足互斥需求。这种方式最大好处就是可以运用与分布式环境。说到消息,抽象地看有两种操作方式:send和receive。从同步方式上看分为阻塞和非阻塞两种,其组合起来有以下 情况:

阻塞send,阻塞receive。发送进程和接收进程都被阻塞,直到信息交付,同步性最好
非阻塞send,阻塞receive。最为自然的一对组合
非阻塞send,非阻塞receive。
那么通过实现以上send和receive原语操作,就可达到互斥的目的,以下面伪代码为例,其中receive为阻塞的,send为非阻塞的:

void onlyOneThread(){  
    receive(box,msg);  
    /*临界区*/ 
    send(box,msg);  


小结

以上是从操作系统的底层来看待并发问题,平常的开发过程一般不需要了解,但透过其原理,我们可以发掘一些解决并发问题的思路。只有真正了解并发产生的原因和操作系统采取的办法,我们才能理解在更高一个层次(比如高级语言编程)为什么有那些控制和措施,为什么对一些代码要做并发控制。


本文来自CSDN博客,转载请标明出处:http://blog.csdn.net/cutesource/archive/2010/07/31/5778820.aspx
分享到:
评论

相关推荐

    Java并发编程实践-电子书

    Java并发编程实践-电子书-01章.pdf Java并发编程实践-电子书-02章.pdf Java并发编程实践-电子书-03章.pdf Java并发编程实践-电子书-04章.pdf Java并发编程实践-电子书-05章.pdf Java并发编程实践-电子书-06章.pdf ...

    62-Java并发编程实战

    62-Java并发编程实战62-Java并发编程实战62-Java并发编程实战62-Java并发编程实战62-Java并发编程实战62-Java并发编程实战62-Java并发编程实战62-Java并发编程实战62-Java并发编程实战62-Java并发编程实战62-Java...

    Java并发编程---synchronized关键字

    Java并发编程---synchronized关键

    JAVA并发编程实践-中文-高清-带书签-完整版

    《JAVA并发编程实践》是Java开发人员深入理解并发编程的一本经典著作,由Doug Lea撰写,本书中文版高清完整,包含丰富的书签,便于读者查阅和学习。这本书旨在帮助开发者掌握在Java平台上进行高效、安全并发编程的...

    Java并发编程实践-电子书-07章

    深入地探讨Java并发编程实践中的显示锁概念,我们聚焦于《Java并发编程实践-电子书-07章》所提供的丰富内容。本章着重于展示如何使用显示锁(也称为高级锁),并详细介绍了Lock和ReentrantLock接口以及它们在并发...

    Java并发编程实践-电子书1-9章pdf

    9. **第九章:实战与性能调优** - 最后一章可能包括了实际并发编程中的最佳实践,性能测试和监控,以及如何根据系统需求进行性能调优。 以上只是基于常规并发编程书籍结构的猜测,具体内容还需要参考实际的电子书...

    go 并发编程实战-郝林

    《Go并发编程实战》是郝林撰写的一本深入解析Go语言并发编程的书籍,它以其深入浅出的讲解和丰富的实战示例深受程序员喜爱。在Go语言中,并发编程是其核心特性之一,也是实现高性能服务的关键技术。下面将详细讨论这...

    Java并发编程实践-电子书-03章

    `java.util.concurrent`包是在JDK5.0之后引入的,它为多线程编程提供了强大的支持,旨在更好地利用现代多处理器或多核系统的性能,以便编写大规模的并发应用程序。这一章节首先介绍了`java.util.concurrent`包的基本...

    Java并发编程实践-电子书-09章.pdf

    Java并发编程实践-电子书-09章.pdf

    高并发编程资料-java

    高并发编程资料-springboot、高并发 适用于学生及工作后的人员

    java并发编程1-9

    java并发编程1-9,可解压,并发编程必看资料。 1 Java 并发编程实践基础 2 构建线程安全应用程序 3 使用JDK 并发包构建程序 4 使用开源软件 Amino 构建并发应用程序 5 数据冲突及诊断工具MTRAT 6 死锁 7 显示锁 ...

    JAVA并发编程实践-

    根据提供的信息,我们可以推断出该资源主要关注的是“Java并发编程实践”的相关内容,并且它是一本高清晰度的PDF电子书。虽然提供的链接部分似乎只是重复的网站地址,我们仍可以根据标题、描述以及标签来生成相关的...

    阿里专家级并发编程架构师课程 彻底解决JAVA并发编程疑难杂症 JAVA并发编程高级教程

    阿里专家级并发编程架构师级课程,完成课程的学习可以帮助同学们解决非常多的JAVA并发编程疑难杂症,极大的提高JAVA并发编程的效率。课程内容包括了JAVA手写线程池,UC线程池API详解,线程安全根因详解,锁与原子类...

    java高级技术JUC高并发编程教程2021(1.5G)

    java高级技术JUC高并发编程教程2021(1.5G) 〖课程介绍〗: java高级技术JUC高并发编程教程2021(1.5G) 〖课程目录〗:   01-JUC高并发编程-课程介绍.mp4 02-JUC高并发编程-JUC概述和进程线程概念(1).mp4 03-JUC...

Global site tag (gtag.js) - Google Analytics