在多线程、多处理器甚至是分布式环境的编程时代,并发是一个不可回避的问题,很多程序员一碰到并发二字头皮就发麻,也包括我。既然并发问题摆在面前 一个到无法回避的坎,倒不如拥抱它,把它搞清楚,决心花一定的时间从操作系统底层原理到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);
- }
小结
以上是从操作系统的底层来看待并发问题,平常的开发过程一般不需要了解,但透过其原理,我们可以发掘一些解决并发问题的思路。只有真正了解并发产生 的原因和操作系统采取的办法,我们才能理解在更高一个层次(比如高级语言编程)为什么有那些控制和措施,为什么对一些代码要做并发控制。
相关推荐
《Java并发编程实战》是Java并发编程领域的一本经典著作,它深入浅出地介绍了如何在Java平台上进行高效的多线程编程。这本书的源码提供了丰富的示例,可以帮助读者更好地理解书中的理论知识并将其应用到实际项目中。...
- **线程** 是操作系统分配CPU执行时间的基本单元,每个线程都有自己的程序计数器、虚拟机栈、本地方法栈和一部分堆内存。 - **并发** 指的是多个任务在同一时间段内被处理,而不是顺序执行。在Java中,这通常通过...
并发编程可以用于操作系统、数据库、网络应用、客户端-服务器架构等多个方面。 Linux大神Paul McKenney的作品《深入理解并发编程》对并发编程的各种概念进行了详细介绍。该书深入探讨了内存屏障(Memory Barriers)...
Windows并发编程是指在Windows操作系统平台上进行的并发程序设计和开发。并发编程是一种允许同时执行多个操作的技术,旨在提高程序性能,特别是在多核处理器上。本书《Windows并发编程指南》详细介绍了并发编程在...
- **线程**:线程是操作系统调度的基本单位,一个进程可以包含多个线程,它们共享进程的内存空间,各自拥有独立的执行流。 - **并发与并行**:并发是指多个任务在同一时间段内交替执行,而并行则是在同一时间点上...
并发编程是现代计算机系统中不可或缺的一部分,尤其是在多核处理器成为主流的今天。Java语言提供了丰富的并发工具和API,如线程、守护线程、线程池、同步机制(synchronized、wait/notify)、并发集合...
【深入理解高并发编程-核心技术原理】是一本专注于讲解高并发编程核心概念和技术的书籍,由阿里P8级别的架构师及Mykit系列开源框架作者撰写。本书内容涵盖源码分析、基础案例、实战案例和面试相关知识,旨在帮助读者...
Java并发编程是软件开发中的一个关键领域,尤其是在大型企业级应用和分布式系统中。通过学习相关的书籍,开发者可以深入理解如何有效地设计和实现高效的多线程应用程序,避免并发问题,如竞态条件、死锁、活锁等。...
#### 一、简介与并发编程基础 《C++并发编程实践》这本书由Anthony Williams编写,是一本深入讲解C++多线程编程技术的专业书籍。本书旨在帮助读者掌握C++中的并发编程技巧,并通过大量的示例代码来加深理解。 **...
这份“java并发编程内部分享PPT”显然是一个深入探讨这一主题的资料,旨在帮助开发者理解并掌握Java并发编程的核心概念和技术。 在Java并发编程中,首先我们需要了解的基本概念是线程。线程是操作系统分配CPU时间的...
在深入理解Java内存模型(JMM)及并发三大特性方面,我们需要先建立对多线程、共享内存模型、可见性、...如果对计算机组成原理和操作系统知识感兴趣,可以通过相关课程进行系统性学习,以便对并发编程有更深入的理解。
在本资源中,我们有两个主要的学习材料:一个关于“Java并发编程基础”的PPT和一个包含DEMO示例,另一个是“操作系统概述”的PPT。这些资料对于理解Java多线程编程以及操作系统的基础原理至关重要。 首先,让我们...
综上所述,《Java并发编程实战》不仅涵盖了Java并发编程的基础知识和技术细节,还包含了丰富的实践经验和前瞻性的思考,是任何一位从事Java开发工作的程序员不可或缺的学习资源。无论是初学者还是有经验的开发者都能...
Java并发编程实践是Java开发中不可或缺的一个领域,它涉及到如何高效、正确地处理多线程环境中的任务。这本书的读书笔记涵盖了多个关键知识点,旨在帮助读者深入理解Java并发编程的核心概念。 1. **线程和进程的...
《Java 并发编程实战》是一本专注于Java并发编程的权威指南,对于任何希望深入了解Java多线程和并发控制机制的开发者来说,都是不可或缺的参考资料。这本书深入浅出地介绍了如何在Java环境中有效地管理和控制并发...
在Java编程领域,并发编程是一项核心技能,尤其是在大型系统或分布式应用中,高效地处理多线程和并发操作是至关重要的。"Java并发编程与实践"文档深入剖析了这一主题,旨在帮助开发者理解和掌握如何在Java环境中有效...
在Java中,主要通过线程实现并发,线程是操作系统调度的基本单位,每个线程都有自己的程序计数器、虚拟机栈、本地方法栈和一部分堆内存。 二、并发设计原则 1. 简单性:尽可能保持代码简洁,避免复杂的并发控制结构...
而并发编程则是实现高并发的基础,它涉及到如何在多线程环境下有效地管理计算资源,以提高系统性能和响应速度。 线程安全是并发编程中的核心概念,确保多个线程在共享数据时不会引发错误或不一致状态。线程安全可以...
《Java高并发编程》第一版是一本专注于...通过阅读《Java高并发编程》第一版,开发者不仅可以掌握Java并发编程的核心概念和技术,还能了解到如何在实际工作中设计和实现高效、安全的并发程序,提升系统的并发处理能力。
并发编程是现代多核处理器环境下提升系统性能的关键技术,对于大型应用和高并发系统的开发尤为重要。在Java平台中,掌握并发编程不仅可以提高程序效率,还能有效避免线程安全问题,确保程序的稳定性和可靠性。 1. *...