让我们一起学习管道模式,你会了吗?

开发 前端
管道模式,不属于23种设计模式之一(是责任链模式的一种变体),但是在我们实际业务架构中还是有很多场景适用的,主要用于将复杂的进程分解成多个独立的子任务,像流水线一样去执行,了解一下呗。

[[402569]]

本文转载自微信公众号「JavaKeeper」,作者海星。转载本文请联系JavaKeeper公众号。

管道模式,不属于23种设计模式之一(是责任链模式的一种变体),但是在我们实际业务架构中还是有很多场景适用的,主要用于将复杂的进程分解成多个独立的子任务,像流水线一样去执行,了解一下呗

一、开场

假设我们有这样的一个需求,读取文件内容,并过滤包含 “hello” 的字符串,然后将其反转

Linux 一行搞定

  1. cat hello.txt | grep "hello" | rev 

用世界上最好语言 Java 实现也很简单

  1. File file = new File("/Users/starfish/Documents/hello.txt"); 
  2.  
  3. String content = FileUtils.readFileToString(file,"UTF-8"); 
  4.  
  5. List<String> helloStr = Stream.of(content).filter(s -> s.contains("hello")).collect(Collectors.toList()); 
  6.  
  7. System.out.println(new StringBuilder(String.join("",helloStr)).reverse().toString()); 

再假设我们上边的场景是在一个大型系统中,有这样的数据流需要多次进行复杂的逻辑处理,还是简单粗暴的把一系列流程像上边那样放在一个大组件中吗?

这样的设计完全违背了单一职责原则,我们在增改,或者减少一些处理逻辑的时候,就必须对整个组件进行改动。可扩展性和可重用性几乎没有~~

那有没有一种模式可以将整个处理流程进行详细划分,划分出的每个小模块互相独立且各自负责一小段逻辑处理,这些小模块可以按顺序连起来,前一模块的输出作为后一模块的输入,最后一个模块的输出为最终的处理结果呢?

如此一来修改逻辑时只针对某个模块修改,添加或减少处理逻辑也可细化到某个模块颗粒度,并且每个模块可重复利用,可重用性大大增强。

恩,这就是我们要说的管道模式

二、定义

管道模式(Pipeline Pattern) 是责任链模式(Chain of Responsibility Pattern)的常用变体之一。

顾名思义,管道模式就像一条管道把多个对象连接起来,整体看起来就像若干个阀门嵌套在管道中,而处理逻辑就放在阀门上,需要处理的对象进入管道后,分别经过各个阀门,每个阀门都会对进入的对象进行一些逻辑处理,经过一层层的处理后从管道尾出来,此时的对象就是已完成处理的目标对象。

管道模式用于将复杂的进程分解成多个独立的子任务。每个独立的任务都是可复用的,因此这些任务可以被组合成复杂的进程。

PS:纯的责任链模式在链上只会有一个处理器用于处理数据,而管道模式上多个处理器都会处理数据。

三、角色

管道模式:对于管道模式来说,有 3 个对象:

  • 阀门:处理数据的节点,或者叫过滤器、阶段
  • 管道:组织各个阀门
  • 客户端:构造管道,并调用

四、实例

程序员还是看代码消化才快些,我们用管道模式实现下文章开头的小需求

1、处理器(管道的各个阶段)

  1. public interface Handler<I,O> { 
  2.     O process(I input); 

2、定义具体的处理器(阀门)

  1. public class FileProcessHandler implements Handler<File,String>{ 
  2.  
  3.     @Override 
  4.     public String process(File file) { 
  5.         System.out.println("===文件处理==="); 
  6.         try{ 
  7.             return FileUtils.readFileToString(file,"UTF-8"); 
  8.         }catch (IOException e){ 
  9.             e.printStackTrace(); 
  10.         } 
  11.         return null
  12.     } 
  1. public class CharacterFilterHandler implements Handler<String, String> { 
  2.  
  3.     @Override 
  4.     public String process(String input) { 
  5.         System.out.println("===字符过滤==="); 
  6.         List<String> hello = Stream.of(input).filter(s -> s.contains("hello")).collect(Collectors.toList()); 
  7.         return String.join("",hello); 
  8.     } 
  1. public class CharacterReverseHandler implements Handler<String,String>{ 
  2.  
  3.     @Override 
  4.     public String process(String input) { 
  5.         System.out.println("===反转字符串==="); 
  6.         return new StringBuilder(input).reverse().toString(); 
  7.     } 

3、管道

  1. public class Pipeline<I,O> { 
  2.  
  3.     private final Handler<I,O> currentHandler; 
  4.  
  5.     Pipeline(Handler<I, O> currentHandler) { 
  6.         this.currentHandler = currentHandler; 
  7.     } 
  8.  
  9.     <K> Pipeline<I, K> addHandler(Handler<O, K> newHandler) { 
  10.         return new Pipeline<>(input -> newHandler.process(currentHandler.process(input))); 
  11.     } 
  12.  
  13.     O execute(I input) { 
  14.         return currentHandler.process(input); 
  15.     } 

4、 客户端使用

  1. import lombok.val; 
  2. public class ClientTest { 
  3.  
  4.     public static void main(String[] args) { 
  5.  
  6.         File file = new File("/Users/apple/Documents/hello.txt"); 
  7.  
  8.         val filters = new Pipeline<>(new FileProcessHandler()) 
  9.                 .addHandler(new CharacterFilterHandler()) 
  10.                 .addHandler(new CharacterReverseHandler()); 
  11.         System.out.println(filters.execute(file)); 
  12.     } 

5、结果

UML 类图

产品他么的又来了,这次是删除 hello.txt 中的 world 字符

三下五除二,精通 shell 编程的我搞定了

  1. cat hello.txt |grep hello |rev | tr -d 'world' 

Java 怎么搞,你应该很清晰了吧

五、优缺点

Pipeline 模式的核心思想是将一个任务处理分解为若干个处理阶段(Stage),其中每个处理阶段的输出作为下一个处理阶段的输入,并且各个处理阶段都有相应的工作者线程去执行相应的计算。因此,处理一批任务时,各个任务的各个处理阶段是并行(Parallel)的。通过并行计算,Pipeline 模式使应用程序能够充分利用多核 CPU 资源,提高其计算效率。 ——《Java 多线程编程实战指南》

优点

  • 将复杂的处理流程分解成独立的子任务,解耦上下游处理逻辑,也方便您对每个子任务的测试
  • 被分解的子任务还可以被不同的处理进程复用
  • 在复杂进程中添加、移除和替换子任务非常轻松,对已存在的进程没有任何影响,这就加大了该模式的扩展性和灵活性
  • 对于每个处理单元又可以打补丁,做监听。(这就是切面编程了)

模式需要注意的东西

  1. Pipeline的深度:Pipeline 中 Pipe 的个数被称作 Pipeline 的深度。所以我们在用 Pipeline 的深度与 JVM 宿主机的 CPU 个数间的关系。如果 Pipeline 实例所处的任务多属于 CPU 密集型,那么深度最好不超过 Ncpu。如果 Pipeline 所处理的任务多属于 I/O 密集型,那么 Pipeline 的深度最好不要超过 2*Ncpu。
  2. 基于线程池的 Pipe:如果 Pipe 实例使用线程池,由于有多个 Pipe 实例,更容易出现线程死锁的问题,需要仔细考虑。
  3. 错误处理:Pipe 实例对其任务进行过程中跑出的异常可能需要相应 Pipe 实例之外进行处理。
  4. 此时,处理方法通常有两种:一是各个 Pipe 实例捕获到异常后调用 PipeContext 实例的 handleError 进行错误处理。另一个是创建一个专门负责错我处理的 Pipe 实例,其他 Pipe 实例捕获异常后提交相关数据给该 Pipe 实例处理。
  5. 可配置的 Pipeline:Pipeline 模式可以用代码的方式将若干个 Pipe 实例添加,也可以用配置文件的方式实现动态方式添加 Pipe。

六、Java Function

如果,你的管道逻辑真的很简单,也直接用 Java8 提供的 Function 就,具体实现如下这样

  1. File file = new File("/Users/apple/Documents/hello.txt"); 
  2.  
  3. Function<File,String> readFile = input -> { 
  4.   System.out.println("===文件处理==="); 
  5.   try{ 
  6.     return FileUtils.readFileToString(input,"UTF-8"); 
  7.   }catch (IOException e){ 
  8.     e.printStackTrace(); 
  9.   } 
  10.   return null
  11. }; 
  12.  
  13. Function<String, String> filterCharacter = input -> { 
  14.   System.out.println("===字符过滤==="); 
  15.   List<String> hello = Stream.of(input).filter(s -> s.contains("hello")).collect(Collectors.toList()); 
  16.   return String.join("",hello); 
  17. }; 
  18.  
  19. Function<String, String> reverseCharacter = input -> { 
  20.   System.out.println("===反转字符串==="); 
  21.   return new StringBuilder(input).reverse().toString(); 
  22. }; 
  23.  
  24. final Function<File,String> pipe = readFile 
  25.   .andThen(filterCharacter) 
  26.   .andThen(reverseCharacter); 
  27.  
  28. System.out.println(pipe.apply(file)); 

最后

但是,并不是一碰到这种类似流式处理的任务就需要用管道,Pipeline 模式中各个处理阶段所用的工作者线程或者线程池,表示各个阶段的输入/输出对象的创建和一定(进出队列)都有其自身的时间和空间开销,所以使用 Pipeline 模式的时候需要考虑它所付出的代价。建议处理规模较大的任务,否则可能得不偿失。

参考

https://java-design-patterns.com/patterns/pipeline/

https://developer.aliyun.com/article/778865

https://yasinshaw.com/articles/108

《Java多线程编程实战指南(设计模式篇)》

 

责任编辑:武晓燕 来源: JavaKeeper
相关推荐

2021-12-14 09:34:31

丑数顺序指针

2023-11-13 18:36:04

知识抽取NER

2023-10-31 14:04:17

Rust类型编译器

2022-02-14 10:16:22

Axios接口HTTP

2023-06-07 14:07:00

架构

2023-01-03 08:13:26

GoModulesMaven

2023-04-13 08:40:12

MySQL服务器SELECT

2022-03-31 18:59:43

数据库InnoDBMySQL

2022-03-08 17:52:58

TCP格式IP

2021-08-27 07:06:10

IOJava抽象

2021-12-29 08:27:05

ByteBuffer磁盘服务器

2021-03-18 00:04:13

C# 类型数据

2021-11-26 07:00:05

反转整数数字

2021-07-15 07:23:28

Singlefligh设计

2022-06-26 09:40:55

Django框架服务

2022-02-14 07:03:31

网站安全MFA

2016-09-06 10:39:30

Dell Techno

2009-11-13 14:52:24

ADO.NET XML

2023-08-14 08:38:26

反射reflect结构体

2022-06-15 08:00:50

磁盘RedisRocketMQ
点赞
收藏

51CTO技术栈公众号