图解多线程中的生产者与消费者问题

问题背景

学过操作系统的我们一定都听老师讲过一个经典的问题——生产者消费者问题。这一节我们就借助Java中的多线程来理解一下这个问题。


问题描述

假定我们定义一个生产者和一个消费者,生产者生产一个物品放入仓库,消费者从仓库消费一个物品。

下面我们通过代码简单演示一下这个过程,我们新建示例代码如下:
图解多线程中的生产者与消费者问题_第1张图片
高清图片,打开看代码,节省文章位置!
图解多线程中的生产者与消费者问题_第2张图片

运行程序代码时,你会发现有可能出现下面的这些情况:

这些情况都是在我们模拟的场景中不应该出现的,下面我们分析产生这个问题的原因。

图解多线程中的生产者与消费者问题_第3张图片

原因分析:

产生的原因其实还是由于上一节的多线程的问题造成的,因为线程占用CPU的资源是随机的。具体的请看上一节的开始讲到的。下面我们使用流程图来分析问题:

未对生产和消费过程进行干涉的过程:
图解多线程中的生产者与消费者问题_第4张图片

对生产和消费过程添加互斥锁:
图解多线程中的生产者与消费者问题_第5张图片
这时候我们运行程序,此时又会产生另一个问题,我们的程序会卡在某一个位置不动,我们称这种情况为死锁。

计算机系统中,如果系统的资源分配策略不当,更常见的可能是程序员写的程序有错误等,则会导致进程因竞争资源不当而产生死锁的现象。

在两个或多个任务中,如果每个任务锁定了其他任务试图锁定的资源,此时会造成这些任务永久阻塞,从而出现死锁。例如:事务A 获取了行 1 的共享锁。事务 B 获取了行 2 的共享锁。


解决方案:

在我们的程序中就是因为我们引入互斥锁导致了生产者和消费者竞争资源,解决的办法很简单,我们使用notifyAll();在生产者生产完成和消费者消费完成后唤醒所有的线程。
图解多线程中的生产者与消费者问题_第6张图片


解决问题:

图解多线程中的生产者与消费者问题_第7张图片

总结;

看到这可能很多人觉得我这讲了跟没讲一样,其实不是,因为学过的都知道,操作系统的知识老师在课上讲解的时候都是再说文字,对于这个概念来说大多数人都是一知半解的,这里我没有用一些刻板的文字去解释,而是选择一个经典案例,结合简单的代码复现和流程分析,将抽象的问题形象化,希望大家能通过这些理解生产者消费者问题。

你可能感兴趣的:(日常,后端学习,Java笔记,java,经验分享,程序人生)