生产者和消费者问题
- 格式:doc
- 大小:331.50 KB
- 文档页数:11
生产者与消费者问题实验报告篇一:生产者和消费者问题实验报告实验报告课程名称:操作系统实验名称:生产者和消费者问题学号:学生姓名:班级:指导教师:评分:实验日期:XX年 10月 22 日篇二:操作系统实验报告经典的生产者—消费者问题实验二经典的生产者—消费者问题一、目的实现对经典的生产者—消费者问题的模拟,以便更好的理解经典进程同步问题。
二、实验内容及要求编制生产者—消费者算法,模拟一个生产者、一个消费者,共享一个缓冲池的情形。
1、实现对经典的生产者—消费者问题的模拟,以便更好的理解此经典进程同步问题。
生产者-消费者问题是典型的PV操作问题,假设系统中有一个比较大的缓冲池,生产者的任务是只要缓冲池未满就可以将生产出的产品放入其中,而消费者的任务是只要缓冲池未空就可以从缓冲池中拿走产品。
缓冲池被占用时,任何进程都不能访问。
2、每一个生产者都要把自己生产的产品放入缓冲池,每个消费者从缓冲池中取走产品消费。
在这种情况下,生产者消费者进程同步,因为只有通过互通消息才知道是否能存入产品或者取走产品。
他们之间也存在互斥,即生产者消费者必须互斥访问缓冲池,即不能有两个以上的进程同时进行。
三、生产者和消费者原理分析在同一个进程地址空间内执行两个线程。
生产者线程生产物品,然后将物品放置在一个空缓冲区中供消费者线程消费。
消费者线程从缓冲区中获得物品,然后释放缓冲区。
当生产者线程生产物品时,如果没有空缓冲区可用,那么生产者线程必须等待消费者线程释放一个空缓冲区。
当消费者线程消费物品时,如果没有满的缓冲区,那么消费者线程将被阻挡,直到新的物品被生产出来。
四、生产者与消费者功能描述:生产者功能描述:在同一个进程地址空间内执行两个线程。
生产者线程生产物品,然后将物品放置在一个空缓冲区中供消费者线程消费。
当生产者线程生产物品时,如果没有空缓冲区可用,那么生产者线程必须等待消费者线程释放出一个空缓冲区。
消费者功能描述:消费者线程从缓冲区获得物品,然后释放缓冲区,当消费者线程消费物品时,如果没有满的缓冲区,那么消费者线程将被阻塞,直到新的物品被生产出来。
消费者剩余和生产者剩余例题
消费者剩余和生产者剩余是微观经济学中的重要概念,用来衡
量市场交易对消费者和生产者的福利影响。
消费者剩余是指消费者
愿意为一定数量的产品支付的价格与其实际支付的价格之间的差额,代表了消费者因低于其愿意支付的价格而获得的福利。
生产者剩余
则是指生产者愿意以一定价格出售产品与其实际获得的价格之间的
差额,代表了生产者因高于其愿意接受的价格而获得的福利。
举个例子来说明消费者剩余和生产者剩余,假设某商品的市场
均衡价格为10元,某消费者愿意为该商品支付15元,但实际只需
支付10元购买到了该商品,那么该消费者的消费者剩余为15元-10
元=5元。
而在生产者方面,假设生产者愿意以5元的价格出售该商品,但实际可以以10元的价格出售,那么生产者的生产者剩余为
10元-5元=5元。
从消费者和生产者的角度看,消费者剩余和生产者剩余都是正
面的福利,代表了他们因市场交易而获得的好处。
消费者剩余衡量
了消费者愿意为产品付出的额外福利,而生产者剩余则衡量了生产
者以低于其愿意接受的价格出售产品所获得的额外福利。
总的来说,消费者剩余和生产者剩余的概念有助于我们理解市场交易对消费者和生产者福利的影响,也有助于政府和企业制定相关政策和战略。
希望这个例子能够帮助你更好地理解消费者剩余和生产者剩余的概念。
⽣产者与消费者问题⽣产者与消费者问题是Java多线程中⼀道⾮常经典的问题,问题如下: ⽣产者与消费者问题也称缓存问题,⽣产者与消费者即Java 中的线程,⽣产者与消费者问题即⽣产者⽣产⼀定数量的线程放⼊缓存区中,供消费者消费者消费,在消费和⽣产的过程中,如果⽣产者⽣产的产品超过了缓存区的上限则停⽌⽣产,等待消费者消费,如果缓存区的产品被消费完,消费者则停⽌消费,等待⽣产者⽣产 ⾸先,我们来看题⽬,从题⽬中我们⼀个可以抽取出⼏个实体类呢?答案是4个 Consumer(消费者),Producer(⽣产者),Product(产品),WareHouse(缓冲区,也叫仓库),于是项⽬结构如下,main 为测试类产品类package ProducersAndConsumers;//产品public class Product {//产品需要⼀个id 来表明产品的唯⼀性private Integer productId;//id直接由构造⽅法传⼊public Product(Integer productId) {this.productId = productId;}public Integer getProductId() {return productId;}@Overridepublic String toString() {return "Product{" +"productId=" + productId +'}';}}仓库package ProducersAndConsumers;import java.util.LinkedList;//仓库类public class WareHouse {//仓库容量,我们设置为10个private final int max = 10;//仓库基础的数量private final int base = 0;//我们设置⼀个集合来存放⽣产的产品,由于我们需要⼀个可以弹出最后⼀个产品的⽅法,所以我们在这⾥使⽤LinkedListprivate LinkedList<Product> products = new LinkedList<>();//⽣产⽅法public synchronized void push(Product product) {//判断是否有空间存放产品while(max==products.size()){try{System.out.println("仓库已满,消费者快来消费"+Thread.currentThread().getName()+"停⽌⽣产");//仓库满后停⽌当前线程this.wait();}catch (Exception ex){ex.printStackTrace();}}//⽣产商品products.addLast(product);System.out.println(Thread.currentThread().getName()+"⽣产了⼀个产品:"+product.getProductId()+"号");try{//等待1秒,⽅⾯我们观察Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}notifyAll();}//消费⽅法public synchronized void pop() {//判断是否有产品while (products.size()==base){try{System.out.println("仓库空了,⽣产者快点⽣产"+Thread.currentThread().getName()+"停⽌消费");//仓库空后停⽌当前线程this.wait();} catch (InterruptedException e) {e.printStackTrace();}}//消费商品System.out.println(Thread.currentThread().getName()+"消费了⼀个产品:"+products.getLast().getProductId()+"号"); products.removeLast();try{//等待1秒,⽅⾯我们观察Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}notifyAll();}}⽣产者package ProducersAndConsumers;//⽣产者public class Producer implements Runnable {//⽣产产品的idprivate int count = 0;//仓库private WareHouse wareHouse;//⽣产者和消费者都是⽤同⼀个仓库,所以我们只要声明⼀个仓库,在由构造⽅法传⼊即可public Producer(WareHouse wareHouse) {this.wareHouse = wareHouse;}//⽣产⽅法@Overridepublic void run() {while (true){Product product = new Product(count);wareHouse.push(product);// 产品id不可重复,所以我们使⽤⾃增策略count++;}}}消费者package ProducersAndConsumers;public class Consumer implements Runnable{//仓库private WareHouse wareHouse;//⽣产者和消费者都是⽤同⼀个仓库,所以我们只要声明⼀个仓库,在由构造⽅法传⼊即可public Consumer(WareHouse wareHouse) {this.wareHouse = wareHouse;}//消费⽅法@Overridepublic void run() {while (true){wareHouse.pop();}}}最后测试类package ProducersAndConsumers;//测试类public class Main {public static void main(String[] args) {WareHouse wareHouse = new WareHouse();Producer producer = new Producer(wareHouse);Consumer consumer = new Consumer(wareHouse); Thread producerT = new Thread(producer,"⽣产者"); Thread consumerT = new Thread(consumer,"消费者"); producerT.start();consumerT.start();}}。
生产者消费者问题例题及详解生产者消费者问题是一个经典的并发问题,涉及到两个独立的线程:生产者和消费者。
生产者生产物品,消费者消费物品。
生产者、消费者共享一个公共的固定大小的缓冲区。
以下是一个简单的生产者消费者问题的例子:假设有一个固定大小的缓冲区,大小为N。
生产者负责生成数据放入缓冲区,而消费者负责从缓冲区取出数据并处理。
1. 当缓冲区为空时,消费者被阻塞,等待生产者生产数据。
2. 当缓冲区满时,生产者被阻塞,等待消费者消费数据。
3. 缓冲区的每个元素只能被消费一次。
4. 缓冲区是循环使用的,即当缓冲区的最后一个元素被消费后,下一个元素将是缓冲区的第一个元素。
问题:如何实现这个生产者消费者模型?解答:可以使用条件变量和互斥锁来实现这个模型。
首先,定义一个缓冲区数组和一个计数器变量来跟踪缓冲区的使用情况。
然后,定义两个条件变量:一个用于生产者等待缓冲区非空,另一个用于消费者等待缓冲区非空。
最后,使用互斥锁来保护对缓冲区和计数器的访问。
以下是使用C++实现的代码示例:```cppinclude <iostream>include <thread>include <mutex>include <condition_variable>const int N = 5; // 缓冲区大小int buffer[N]; // 缓冲区数组int count = 0; // 计数器变量,表示缓冲区的使用情况std::mutex mutex; // 互斥锁std::condition_variable cv_prod; // 生产者等待条件变量std::condition_variable cv_cons; // 消费者等待条件变量void producer() {for (int i = 0; i < N 2; i++) {std::unique_lock<std::mutex> lock(mutex);cv_(lock, []{ return count < N; }); // 等待缓冲区非空buffer[count] = i; // 生产数据放入缓冲区std::cout << "Producer produced " << i << std::endl;count++; // 更新计数器变量if (count == N) count = 0; // 循环使用缓冲区cv__one(); // 通知消费者消费数据}}void consumer() {for (int i = 0; i < N 2; i++) {std::unique_lock<std::mutex> lock(mutex);cv_(lock, []{ return count > 0; }); // 等待缓冲区非空int data = buffer[count]; // 从缓冲区取出数据并处理 std::cout << "Consumer consumed " << data << std::endl;count--; // 更新计数器变量if (count == -1) count = N - 1; // 循环使用缓冲区cv__one(); // 通知生产者生产数据}}int main() {std::thread prod(producer); // 创建生产者线程 std::thread cons(consumer); // 创建消费者线程 (); // 等待生产者线程结束(); // 等待消费者线程结束return 0;}```。
操作系统中的经典问题——⽣产者消费者问题(两种⽅式实现)操作系统中的经典问题——⽣产者消费者问题(两种⽅式实现)1、问题引⼊:什么是⽣产者消费者问题?⽣产者消费者问题(英语:Producer-consumer problem),也称有限缓冲问题(英语:Bounded-buffer problem),是⼀个多线程同步问题的经典案例。
该问题描述了共享固定⼤⼩缓冲区的两个线程——即所谓的“⽣产者”和“消费者”——在实际运⾏时会发⽣的问题。
⽣产者的主要作⽤是⽣成⼀定量的数据放到缓冲区中,然后重复此过程。
与此同时,消费者也在缓冲区消耗这些数据。
该问题的关键就是要保证⽣产者不会在缓冲区满时加⼊数据,消费者也不会在缓冲区中空时消耗数据。
.要解决该问题,就必须让⽣产者在缓冲区满时休眠(要么⼲脆就放弃数据),等到下次消费者消耗缓冲区中的数据的时候,⽣产者才能被唤醒,开始往缓冲区添加数据。
同样,也可以让消费者在缓冲区空时进⼊休眠,等到⽣产者往缓冲区添加数据之后,再唤醒消费者。
通常采⽤进程间通信的⽅法解决该问题。
如果解决⽅法不够完善,则容易出现死锁的情况。
出现死锁时,两个线程都会陷⼊休眠,等待对⽅唤醒⾃⼰。
该问题也能被推⼴到多个⽣产者和消费者的情形。
2、问题分析该问题需要注意的⼏点:1. 在缓冲区为空时,消费者不能再进⾏消费2. 在缓冲区为满时,⽣产者不能再进⾏⽣产3. 在⼀个线程进⾏⽣产或消费时,其余线程不能再进⾏⽣产或消费等操作,即保持线程间的同步4. 注意条件变量与互斥锁的顺序由于前两点原因,因此需要保持线程间的同步,即⼀个线程消费(或⽣产)完,其他线程才能进⾏竞争CPU,获得消费(或⽣产)的机会。
对于这⼀点,可以使⽤条件变量进⾏线程间的同步:⽣产者线程在product之前,需要wait直⾄获取⾃⼰所需的信号量之后,才会进⾏product的操作;同样,对于消费者线程,在consume之前需要wait直到没有线程在访问共享区(缓冲区),再进⾏consume的操作,之后再解锁并唤醒其他可⽤阻塞线程。
操作系统课程设计实验报告实验名称: 生产者消费者问题姓名/学号:一、实验目的以生产者和消费者问题为例, 学习Linux和Windows下进程通信、同步机制的具体实现方法, 主要是信号量和共享内存。
熟悉相关系统API的用法。
二、实验内容使用共享内存和信号量机制来实现多个生产者/消费者进程间的通信和同步。
要求在Linux和Windows下分别实现。
缓冲区大小为3, 初始为空。
2个生产者, 随机等待一段时间, 往缓冲区添加数据, 重复6次。
3个消费者, 重复4次。
三、实验环境Ubuntu 10.10 , GCC; Windows 7, VC 6.0;四、程序设计与实现1.Linux下:(1) 数据结构:a.共享内存定义为一个结构, 使得其数据成员更清晰且操作变得简单。
b.共享缓冲区采用循环队列的数据结构,由上面的结构struct buf { int start; int end; int info[BUF_NUM]; }维护。
其中start为队头指针, end为队尾指针, info为数据区域。
(2) 算法:a.大致由三个模块组成:i.主程序(main):ii.创建信号量、共享内存并进行初始化iii.创建生产者、消费者进程, 生产者执行pro_fun(), 消费者执行con_fun()iv.等待所有子进程的结束v.删除信号量、共享内存i.生产者进程(pro_fun):ii.通过key获得信号量、共享内存的ID, 将内存添加到自己的地址空间iii.P(empty), P(mutex), Add(data), V(mutex), V(full)iv.解除和共享内存的关联i.消费者进程(con_fun):ii.通过key获得信号量、共享内存的ID, 将内存添加到自己的地址空间iii.P(full), P(mutex), Add(data), V(mutex), V(empty)iv.解除和共享内存的关联循环队列部分:加入数据: info[end] = value; end = (end + 1) % 3;取出数据: temp = info[start]; info[start] = 0; (start = start + 1)%3; return temp;(3) 程序流程图:a.主函数:b.生产者进程:c.消费者进程和生产者类似4.Windows 下:(1) 数据结构:和Linux大致相同(2) 算法:a.创建的子进程调用正在执行的文件本身, 通过main函数的参数区分主进程和生产者、消费者进程。
第一章、概述1.1 课题背景在多道程序环境下,进程同步问题十分重要,也是一个相当有趣的问题,因而吸引了不少学者对它进行研究,并由此而产生了一系列经典的进程同步问题。
其中比较有代表性的有“生产者—消费者问题” 、“读者—写者问题” 、“哲学家进餐问题”等等。
通过对这些问题的研究和学习,可以帮助我们更好地理解进程同步概念及实现方法。
1.2生产者—消费者问题生产者—消费者问题(Producer_consumer)是一个经典的进程同步问题。
它描述的是:有一群生产者进程在生产产品,并将此产品提供给消费者进程去消费。
为使生产者进程和消费者进程能并发执行,在它们之间设置有个缓冲区的缓冲池,生产者进程可将它所生产的产品放入一个缓冲区中,消费者进程可从一个缓冲区取得一个产品消费。
尽管所有的生产者进程和消费者进程都是以异步的方式运行的,但它们之间必须保持同步,即不允许消费者进程到一个空缓冲区去取产品,也不允许生产者进程向一个已装有消息尚未被取走产品的缓冲区投放产品。
如下图所示:1.3进程同步机制在中引入进程后,虽然提高了资源的利用率和系统的吞吐量,但由于进程的异步性,也会给系统造成混乱,尤其是在它们争用临界资源的时候。
例如,当多个进程去争用一台打印机时,有可能使多个进程的输出结果交织在一起,难于区分;而当多个进程去争用共享变量,表格,链表时,有可能使数据处理出错。
进程同步的主要任务就是使并发执行的诸进程之间能有效地共享资源和相互合作,从而使程序的执行具有可再现性。
1.4进程同步优点进程同步其优点在于能够让操作系统更加有效地对资源进行管理和调度,最大潜力地发挥处理机的性能。
让系统的执行更加畅通无阻,尽可能地让系统少出现一些由于系统资源分配不合理所带来的死锁、死机之类的事情的发生。
保持了处理机的高速运行之后从用户角度来说程序运行所花费的时间就会更短。
从而保证了处理机在相同的时间内有更大的吞吐量。
而把并发进程的同步和互斥问题一般化,就可以得到一个抽象的一般模型,即本次课程设计的任务:生产者—消费者问题。
生产者消费者问题实验报告生产者消费者问题实验报告一、引言生产者消费者问题是计算机科学中一个经典的并发问题,主要涉及到多个线程之间的协作和资源的共享。
在本实验中,我们通过编写一个简单的程序来模拟生产者和消费者之间的交互过程,以深入理解该问题的本质和解决方案。
二、问题描述在生产者消费者问题中,有两类线程:生产者和消费者。
生产者线程负责生产一定数量的产品,而消费者线程则负责消费这些产品。
两类线程需要共享一个有限的缓冲区,生产者将产品放入缓冲区,而消费者从缓冲区中取出产品。
然而,缓冲区的容量是有限的,当缓冲区已满时,生产者需要等待,直到有空间可用。
同样地,当缓冲区为空时,消费者需要等待,直到有产品可用。
三、实验设计为了解决生产者消费者问题,我们采用了经典的解决方案——使用互斥锁和条件变量。
互斥锁用于保护共享资源的访问,保证同一时间只有一个线程可以访问共享资源。
而条件变量用于线程之间的通信,当某个条件不满足时,线程可以通过条件变量进入等待状态,直到条件满足时再被唤醒。
在我们的程序中,我们使用了一个有界缓冲区来模拟生产者消费者之间的交互。
缓冲区的大小可以通过参数进行设置。
我们创建了两个线程分别代表生产者和消费者,它们通过互斥锁和条件变量来实现同步。
生产者线程在缓冲区未满时将产品放入缓冲区,并通知消费者线程有产品可用;消费者线程在缓冲区非空时从缓冲区取出产品,并通知生产者线程有空间可用。
通过这种方式,我们保证了生产者和消费者之间的协作和资源的共享。
四、实验结果经过多次运行实验,我们观察到了以下现象:当生产者线程的生产速度大于消费者线程的消费速度时,缓冲区会被生产者填满,消费者需要等待;当消费者线程的消费速度大于生产者线程的生产速度时,缓冲区会被消费者清空,生产者需要等待。
只有当生产者和消费者的速度相等时,才能实现平衡的生产和消费。
此外,我们还发现在某些情况下,生产者和消费者线程可能出现死锁或饥饿现象。
死锁是指两个或多个线程相互等待对方释放资源,导致程序无法继续执行的情况。
⽣产者消费者问题⽣产者消费者问题背景在并发编程中,⽣产者消费者问题(producer/consumer)是⼀个经典的⽼⽣常谈的问题,有时也称为有界缓冲区问题。
问题的基本背景假设是:我们有⼀个固定⼤⼩的缓冲区,这个缓冲区分别有两种⼯作性质不同的线程去操作。
其中⼀种线程负责向缓冲区中写⼊数据,我们称之为⽣产者线程。
另⼀种线程则负责从缓冲区中拿取数据,并称之为消费者线程。
同时两种线程的写⼊和拿取⼯作要遵循⼀定的规则:1. 缓冲区未写满时,⽣产者线程可以向缓冲区中写⼊数据。
但是消费者线程不能从缓冲区中读取数据。
2. 缓冲区写满时,⽣产者线程不能向缓冲区中写⼊数据,消费者线程可以冲缓冲区中读取数据。
3. 不管是那种性质的线程,在操作缓冲区时,均不可出现并发安全问题。
分析可以得知,解决⽣产者消费者问题,其实就是要解决线程同步问题与共享资源互斥访问问题。
互斥问题的解决可以借助锁来实现,⽽线程同步则需借助信号量或其他⼯具来实现。
Java实现class FixedSizeBuffer{private static final int DEFAULT_BUFFER_SIZE = 1024;private final ReentrantLock lock = new ReentrantLock(); // 共享资源访问锁private final Condition isFull = lock.newCondition(); // buffer是否已满private final Condition isEmpty = lock.newCondition(); // buffer是否还空着private final int size; // buffer的⼤⼩private final byte[] buffer; // bufferprivate int cursor; // 写⼊游标public FixedSizeBuffer(){this(DEFAULT_BUFFER_SIZE);}public FixedSizeBuffer(int size){if (size <= 0) throw new IllegalArgumentException();this.size = size;this.buffer = new byte[size];cursor = -1;}/*** 向buffer中写⼊⼀个字节的数据* @param content 数据内容* @throws InterruptedException 中断异常*/public void putByte(byte content) throws InterruptedException{/*由于要对共享资源buffer进⾏访问,所以要加锁。
课程设计题目生产者和消费者问题学院计算机科学与技术专业班级姓名指导教师吴利军2013 年 1 月16 日课程设计任务书学生姓名:指导教师:吴利军工作单位:计算机科学与技术学院题目: 进程同步模拟设计——生产者和消费者问题初始条件:1.预备内容:阅读操作系统的进程管理章节内容,对进程的同步和互斥,以及信号量机制度有深入的理解。
2.实践准备:掌握一种计算机高级语言的使用。
要求完成的主要任务:(包括课程设计工作量及其技术要求,以及说明书撰写等具体要求)1.模拟用信号量机制实现生产者和消费者问题。
2.设计报告内容应说明:⑴需求分析;⑵功能设计(数据结构及模块说明);⑶开发平台及源程序的主要部分;⑷测试用例,运行结果与运行情况分析;⑸自我评价与总结:i)你认为你完成的设计哪些地方做得比较好或比较出色;ii)什么地方做得不太好,以后如何改正;iii)从本设计得到的收获(在编写,调试,执行过程中的经验和教训);iv)完成本题是否有其他方法(如果有,简要说明该方法);时间安排:设计安排一周:周1、周2:完成程序分析及设计。
周2、周3:完成程序调试及测试。
周4、周5:验收、撰写课程设计报告。
(注意事项:严禁抄袭,一旦发现,一律按0分记)指导教师签名:年月日系主任(或责任教师)签名:年月日生产者-消费者问题(the producer-consumer problem) 1.需求分析1.1问题描述:一组生产者向一组消费者提供消息,它们共享一个有界缓冲区n,生产者向其中投放消息,消费者从中取得消息。
1.2规则:⏹对于生产者进程:产生一个数据,当要送入缓冲区时,要检查缓冲区是否已满,若未满,则可将数据送入缓冲区,并通知消费者进程;否则,等待;⏹对于消费者进程:当它去取数据时,要看缓冲区中是否有数据可取,若有则取走一个数据,并通知生产者进程,否则,等待。
⏹缓冲区是个临界资源,因此,诸进程对缓冲区的操作程序是一个共享临界区,所以,还有个互斥的问题。
1.3信号灯设置:两个同步信号灯--empty :表示空缓冲区的数目,初值为有界缓冲区的大小n;full :表示满缓冲区(即信息)的数目,其初值为0;一个互斥信号灯--mutex:互斥信号灯,初值为1。
1.4同步描述:1.5程序描述:main( ){int full=0;/* 满缓冲区的数目 */int empty=n;/* 空缓冲区的数目 */int mutex=1;/* 对有界缓冲区进行操作的互斥信号灯*/cobeginp1 ( );p2( );coend}p1(){while(生产未完成){···生产一个产品;p(empty);p(mutex);送一个产品到有界缓冲区;v(mutex);v(full);}}p2( ){while(还要继续消费){p(full);p(mutex);从有界缓冲区中取产品;v(mutex);v(empty);···消费一个产品;}}1.6C++语言程序模拟用信号量机制实现生产者和消费者问题:本次课程设计主要通过C++模拟信号量制中各个进程,及各进程之间的互斥、同步关系,来实现生产者和消费者问题。
2.功能设计2.1设计目的:通过实验模拟生产者和消费者之间的关系,了解并掌握他们之间的关系及其原理。
由此增加对进程同步的问题的了解。
具体如下:1)掌握基本的同步互斥算法,理解生产者和消费者模型;2)了解windows中多线程(多进程)的并发执行机制,线程(进程)间的同步和互斥; 3)学习使用windows中基本的同步对象,掌握相应的API。
2.2设计功能:利用模拟用信号量机制实现生产者和消费者问题:通过用户控制取进程和放进程,反应生产者和消费者问题中所涉及的进程的同步与互斥。
2.3数据结构:2.4模块说明:const unsigned short SIZE_OF_BUFFER = 10; //缓冲区长度unsigned short ProductID = 0; //产品号unsigned short ConsumeID = 0; //将被消耗的产品号unsigned short in = 0; //产品进缓冲区时的缓冲区下标unsigned short out = 0; //产品出缓冲区时的缓冲区下标int g_buffer[SIZE_OF_BUFFER]; //缓冲区是个循环队列bool g_continue = true; //控制程序结束HANDLE g_hMutex; //用于线程间的互斥HANDLE g_hFullSemaphore; //当缓冲区满时迫使生产者等待HANDLE g_hEmptySemaphore; //当缓冲区空时迫使消费者等待DWORD WINAPI Producer(LPVOID); //生产者线程DWORD WINAPI Consumer(LPVOID); //消费者线程3.开发平台及源程序的主要部分:3.1开发平台:基于VS2010开发平台的C++编程3.2源程序的主要部分:#include <windows.h>#include <iostream>const unsigned short SIZE_OF_BUFFER = 10; //缓冲区长度unsigned short ProductID = 0; //产品号unsigned short ConsumeID = 0; //将被消耗的产品号unsigned short in = 0; //产品进缓冲区时的缓冲区下标unsigned short out = 0; //产品出缓冲区时的缓冲区下标int g_buffer[SIZE_OF_BUFFER]; //缓冲区是个循环队列bool g_continue = true; //控制程序结束HANDLE g_hMutex; //用于线程间的互斥HANDLE g_hFullSemaphore; //当缓冲区满时迫使生产者等待HANDLE g_hEmptySemaphore; //当缓冲区空时迫使消费者等待DWORD WINAPI Producer(LPVOID); //生产者线程DWORD WINAPI Consumer(LPVOID); //消费者线程int main(){//创建各个互斥信号g_hMutex = CreateMutex(NULL,FALSE,NULL);g_hFullSemaphore = CreateSemaphore(NULL,SIZE_OF_BUFFER-1,SIZE_OF_BUFFER-1,NULL);g_hEmptySemaphore = CreateSemaphore(NULL,0,SIZE_OF_BUFFER-1,NULL);//调整下面的数值,可以发现,当生产者个数多于消费者个数时,//生产速度快,生产者经常等待消费者;反之,消费者经常等待const unsigned short PRODUCERS_COUNT = 3; //生产者的个数const unsigned short CONSUMERS_COUNT = 1; //消费者的个数//总的线程数const unsigned short THREADS_COUNT = PRODUCERS_COUNT+CONSUMERS_COUNT;HANDLE hThreads[PRODUCERS_COUNT]; //各线程的handleDWORD producerID[CONSUMERS_COUNT]; //生产者线程的标识符DWORD consumerID[THREADS_COUNT]; //消费者线程的标识符//创建生产者线程for (int i=0;i<PRODUCERS_COUNT;++i){hThreads[i]=CreateThread(NULL,0,Producer,NULL,0,&producerID[i]);if (hThreads[i]==NULL) return -1;}//创建消费者线程for (int j=0; j<CONSUMERS_COUNT; ++j){hThreads[PRODUCERS_COUNT+j]=CreateThread(NULL,0,Consumer,NULL,0,&consumerID[j ]);if (hThreads[j]==NULL) return -1;}while(g_continue){if(getchar()){ //按回车后终止程序运行g_continue = false;}}return 0;}//生产一个产品。
简单模拟了一下,仅输出新产品的ID号void Produce(){std::cerr << "Producing " << ++ProductID << " ... "; std::cerr << "Succeed" << std::endl;}//把新生产的产品放入缓冲区void Append(){std::cerr << "Appending a product ... ";g_buffer[in] = ProductID;in = (in+1)%SIZE_OF_BUFFER;std::cerr << "Succeed" << std::endl;//输出缓冲区当前的状态for (int i=0;i<SIZE_OF_BUFFER;++i){std::cout << i <<": " << g_buffer[i];if (i==in) std::cout << " <-- 生产";if (i==out) std::cout << " <-- 消费";std::cout << std::endl;}}//从缓冲区中取出一个产品void Take(){std::cerr << "Taking a product ... ";ConsumeID = g_buffer[out];out = (out+1)%SIZE_OF_BUFFER;std::cerr << "Succeed" << std::endl;//输出缓冲区当前的状态for (int i=0;i<SIZE_OF_BUFFER;++i){std::cout << i <<": " << g_buffer[i];if (i==in) std::cout << " <-- 生产";if (i==out) std::cout << " <-- 消费";std::cout << std::endl;}}//消耗一个产品void Consume(){std::cerr << "Consuming " << ConsumeID << " ... "; std::cerr << "Succeed" << std::endl;}//生产者DWORD WINAPI Producer(LPVOID lpPara){while(g_continue){WaitForSingleObject(g_hFullSemaphore,INFINITE); WaitForSingleObject(g_hMutex,INFINITE);Produce();Append();Sleep(1500);ReleaseMutex(g_hMutex);ReleaseSemaphore(g_hEmptySemaphore,1,NULL);}return 0;}//消费者DWORD WINAPI Consumer(LPVOID lpPara){while(g_continue){WaitForSingleObject(g_hEmptySemaphore,INFINITE); WaitForSingleObject(g_hMutex,INFINITE);Take();Consume();Sleep(1500);ReleaseMutex(g_hMutex);ReleaseSemaphore(g_hFullSemaphore,1,NULL);}return 0;}4.测试用例,运行结果与运行情况分析:4.1测试用例:在本次课程设计的测试中,缓冲区长度是取值为10,生产者个数取值为3,消费者个数取值为1.4.2运行结果与分析:在测试中调整生产者和消费者的数值,可以发现,当生产者个数多于消费者个数时,生产速度快,生产者经常等待消费者;反之,消费者经常等待。