热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

关于java:Java中的设计模式二生产者消费者模式与观察者模式

在上一篇Java中的设计模式(一):观察者模式中咱们理解了观察者模式的基本原理和应用场景,在明天的这篇文章中咱们要做一点简略的延伸性学习——比照一下生产者-消费者模式和观察者模式的异同。

工具与资源核心
帮忙开发者更加高效的工作,提供围绕开发者全生命周期的工具与资源
https://developer.aliyun.com/…

一、前言

  在上一篇 Java中的设计模式(一):观察者模式 中咱们理解了 观察者模式 的基本原理和应用场景,在明天的这篇文章中咱们要做一点简略的延伸性学习——比照一下 生产者-消费者模式 和 观察者模式 的异同。

二、什么是“生产者-消费者模式”?

  和观察者模式不同,生产者-消费者模式 自身并不属于设计模式中的任何一种 。那么生产者-消费者模式到底是什么呢?上面咱们用一个例子简略阐明一下:

  如同上图中所示,生产者和消费者就如同一本杂志的投稿作者和订阅的读者,同一本杂志的投稿作者能够有多个,它的读者也能够有多个,而杂志就是连贯作者和读者的桥梁(即缓冲区)。通过杂志这个数据缓冲区,作者能够将实现的作品投递给订阅了杂志的读者,在这一过程中,作者不必关怀读者是否收到了作品或是否实现了浏览,作者和读者是两个绝对独立的对象,两者的行为互不影响。
  能够看到,在这个例子当中呈现了三个角色,别离是 生产者 、 消费者 以及 缓冲区 。生产者和消费者比拟好了解,前者是生产数据,后者则是解决前者生产进去的数据。而缓冲区在生产者-消费者模式中则起到了一个 解耦 、 反对异步 、 反对忙闲不均 的作用。

三、两者的区别

1. 编程范式不同

  生产者-消费者模式和观察者模式的第一个不同点在下面曾经说过,前者是一种 面向过程 的软件设计模式,不属于Gang of Four提出的23种设计模式中的任何一种,而后者则是23中设计模式中的一种,也即面向对象的设计模式中的一种。

2. 关联关系不同

  这一理念上的不同就带出了下一种不同点,即观察者模式中只有一对多的关系,没有多对多的关系,而在生产者-消费者模式中则是多对多的关系。
  在观察者模式中,被观察者只有一个,观察者却能够有多个。就比方十字路口的交通灯,直行的车辆只会察看管制直行的交通灯,不会去察看管制左拐或者右拐的交通灯,也就是说察看的对象是固定惟一的。
  而在生产者-消费者模式中则不同,生产者能够有多个,消费者也能够有多个。还是用下面作者和读者的例子,在这个例子当中,读者只关怀杂志的内容而不用关怀内容的创作者是谁,作者也只须要晓得创作完的作品能够公布到对应的杂志,而不用关怀会有那些读者。

3. 耦合关系不同

  从上一个不同中不难看出生产者-消费者模式和观察者模式的耦合关系也不雷同,前者为 轻耦合 ,后者为 重耦合 。

4. 利用场景不同

  观察者模式多用于 事件驱动模型 当中,生产者-消费者模式则多呈现在 过程间通信 ,用于进行解耦和并发解决,咱们罕用的音讯队列用的就是生产者-消费者模式。当然在Java中应用生产者-消费者模式还须要留神缓冲区的线程平安问题,这里就不做过多叙述。

四、一个小例子

  最初用一个简略的demo来完结本次的延长学习。

1. StoreQueue–缓冲区

public class StoreQueue {
    private final BlockingQueue queue = new LinkedBlockingQueue<>();
    /**
     * 队列中减少数据
     *
     * @param data 生产者生产的数据
     */
    public void add(T data) {
        try {
            queue.put(data);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
    /**
     * 队列中获取数据
     *
     * @return 从队列中获取到的数据
     */
    public T get() {
        try {
            return queue.take();
        } catch (Exception e) {
            e.printStackTrace();
        }
        return null;
    }
}

  在这个例子中,咱们应用了jdk本身的 阻塞队列BlockingQueue 来实现了一个缓冲区,这里只须要实现放数据和取数据的办法。如果咱们本人实现一个阻塞队列,一方面须要留神阻塞的解决,另一方面须要思考线程平安的问题,这里就不开展叙述了,有趣味的同学能够看下BlockingQueue的源码。

2. Producer&#8211;生产者

public class Producer implements Runnable{
    private StoreQueue storeQueue;
    public Producer(StoreQueue storeQueue) {
        this.storeQueue = storeQueue;
    }
    @Override
    public void run() {
        for (int i = 0; i <10; i++) {
            storeQueue.add(Thread.currentThread().getName() + ":" + i);
        }
    }
}

3. Consumer&#8211;消费者

public class Consumer implements Runnable{
    private StoreQueue storeQueue;
    public Consumer(StoreQueue storeQueue) {
        this.storeQueue = storeQueue;
    }
    @Override
    public void run() {
        try {
            while (true) {
                String data = storeQueue.get();
                System.out.println("以后生产线程 : " + Thread.currentThread().getName() + ", 接管到数据 : " + data);
            }
        } catch (Exception e) {
            e.printStackTrace();
            Thread.currentThread().interrupt();
        }
    }
}

4. 执行逻辑和运行后果

执行逻辑

public static void main(String[] args) {
        StoreQueue storeQueue = new StoreQueue<>();
        Producer producer = new Producer(storeQueue);
        Consumer cOnsumer= new Consumer(storeQueue);
        Producer producerTwo = new Producer(storeQueue);
        Consumer cOnsumerTwo= new Consumer(storeQueue);
        new Thread(producer).start();
        new Thread(consumer).start();
        new Thread(producerTwo).start();
        new Thread(consumerTwo).start();
    }

运行后果

以后生产线程 : Thread-1, 接管到数据 : Thread-0:0
以后生产线程 : Thread-1, 接管到数据 : Thread-0:1
以后生产线程 : Thread-1, 接管到数据 : Thread-0:2
以后生产线程 : Thread-1, 接管到数据 : Thread-0:3
以后生产线程 : Thread-1, 接管到数据 : Thread-0:4
以后生产线程 : Thread-3, 接管到数据 : Thread-0:5
以后生产线程 : Thread-3, 接管到数据 : Thread-0:7
以后生产线程 : Thread-3, 接管到数据 : Thread-0:8
以后生产线程 : Thread-3, 接管到数据 : Thread-0:9
以后生产线程 : Thread-3, 接管到数据 : Thread-2:0
以后生产线程 : Thread-3, 接管到数据 : Thread-2:1
以后生产线程 : Thread-3, 接管到数据 : Thread-2:2
以后生产线程 : Thread-3, 接管到数据 : Thread-2:3
以后生产线程 : Thread-3, 接管到数据 : Thread-2:4
以后生产线程 : Thread-3, 接管到数据 : Thread-2:5
以后生产线程 : Thread-3, 接管到数据 : Thread-2:6
以后生产线程 : Thread-3, 接管到数据 : Thread-2:7
以后生产线程 : Thread-3, 接管到数据 : Thread-2:8
以后生产线程 : Thread-3, 接管到数据 : Thread-2:9
以后生产线程 : Thread-1, 接管到数据 : Thread-0:6

  能够看到在下面的数据后果中,不同生产者生产的数据只会被一个消费者生产,没有呈现线程平安问题,这要归功于实现缓冲区应用到的 BlockingQueue 。

本文转自:https://developer.aliyun.com/&#8230;


推荐阅读
  • Java太阳系小游戏分析和源码详解
    本文介绍了一个基于Java的太阳系小游戏的分析和源码详解。通过对面向对象的知识的学习和实践,作者实现了太阳系各行星绕太阳转的效果。文章详细介绍了游戏的设计思路和源码结构,包括工具类、常量、图片加载、面板等。通过这个小游戏的制作,读者可以巩固和应用所学的知识,如类的继承、方法的重载与重写、多态和封装等。 ... [详细]
  • Iamtryingtomakeaclassthatwillreadatextfileofnamesintoanarray,thenreturnthatarra ... [详细]
  • 本文讨论了一个关于cuowu类的问题,作者在使用cuowu类时遇到了错误提示和使用AdjustmentListener的问题。文章提供了16个解决方案,并给出了两个可能导致错误的原因。 ... [详细]
  • 重入锁(ReentrantLock)学习及实现原理
    本文介绍了重入锁(ReentrantLock)的学习及实现原理。在学习synchronized的基础上,重入锁提供了更多的灵活性和功能。文章详细介绍了重入锁的特性、使用方法和实现原理,并提供了类图和测试代码供读者参考。重入锁支持重入和公平与非公平两种实现方式,通过对比和分析,读者可以更好地理解和应用重入锁。 ... [详细]
  • 本文介绍了闭包的定义和运转机制,重点解释了闭包如何能够接触外部函数的作用域中的变量。通过词法作用域的查找规则,闭包可以访问外部函数的作用域。同时还提到了闭包的作用和影响。 ... [详细]
  • 在Android开发中,使用Picasso库可以实现对网络图片的等比例缩放。本文介绍了使用Picasso库进行图片缩放的方法,并提供了具体的代码实现。通过获取图片的宽高,计算目标宽度和高度,并创建新图实现等比例缩放。 ... [详细]
  • 向QTextEdit拖放文件的方法及实现步骤
    本文介绍了在使用QTextEdit时如何实现拖放文件的功能,包括相关的方法和实现步骤。通过重写dragEnterEvent和dropEvent函数,并结合QMimeData和QUrl等类,可以轻松实现向QTextEdit拖放文件的功能。详细的代码实现和说明可以参考本文提供的示例代码。 ... [详细]
  • Java容器中的compareto方法排序原理解析
    本文从源码解析Java容器中的compareto方法的排序原理,讲解了在使用数组存储数据时的限制以及存储效率的问题。同时提到了Redis的五大数据结构和list、set等知识点,回忆了作者大学时代的Java学习经历。文章以作者做的思维导图作为目录,展示了整个讲解过程。 ... [详细]
  • 本文介绍了OC学习笔记中的@property和@synthesize,包括属性的定义和合成的使用方法。通过示例代码详细讲解了@property和@synthesize的作用和用法。 ... [详细]
  • XML介绍与使用的概述及标签规则
    本文介绍了XML的基本概念和用途,包括XML的可扩展性和标签的自定义特性。同时还详细解释了XML标签的规则,包括标签的尖括号和合法标识符的组成,标签必须成对出现的原则以及特殊标签的使用方法。通过本文的阅读,读者可以对XML的基本知识有一个全面的了解。 ... [详细]
  • 深入理解Kafka服务端请求队列中请求的处理
    本文深入分析了Kafka服务端请求队列中请求的处理过程,详细介绍了请求的封装和放入请求队列的过程,以及处理请求的线程池的创建和容量设置。通过场景分析、图示说明和源码分析,帮助读者更好地理解Kafka服务端的工作原理。 ... [详细]
  • 本文讨论了一个数列求和问题,该数列按照一定规律生成。通过观察数列的规律,我们可以得出求解该问题的算法。具体算法为计算前n项i*f[i]的和,其中f[i]表示数列中有i个数字。根据参考的思路,我们可以将算法的时间复杂度控制在O(n),即计算到5e5即可满足1e9的要求。 ... [详细]
  • 李逍遥寻找仙药的迷阵之旅
    本文讲述了少年李逍遥为了救治婶婶的病情,前往仙灵岛寻找仙药的故事。他需要穿越一个由M×N个方格组成的迷阵,有些方格内有怪物,有些方格是安全的。李逍遥需要避开有怪物的方格,并经过最少的方格,找到仙药。在寻找的过程中,他还会遇到神秘人物。本文提供了一个迷阵样例及李逍遥找到仙药的路线。 ... [详细]
  • 本文介绍了Codeforces Round #321 (Div. 2)比赛中的问题Kefa and Dishes,通过状压和spfa算法解决了这个问题。给定一个有向图,求在不超过m步的情况下,能获得的最大权值和。点不能重复走。文章详细介绍了问题的题意、解题思路和代码实现。 ... [详细]
  • 本文介绍了在Android开发中使用软引用和弱引用的应用。如果一个对象只具有软引用,那么只有在内存不够的情况下才会被回收,可以用来实现内存敏感的高速缓存;而如果一个对象只具有弱引用,不管内存是否足够,都会被垃圾回收器回收。软引用和弱引用还可以与引用队列联合使用,当被引用的对象被回收时,会将引用加入到关联的引用队列中。软引用和弱引用的根本区别在于生命周期的长短,弱引用的对象可能随时被回收,而软引用的对象只有在内存不够时才会被回收。 ... [详细]
author-avatar
黑夜乱来
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有