用Java 8 Lambda表达式实现设计模式:命令模式

链接:http://www.importnew.com/16789.html

在这篇博客里,我将说明如何在使用Java 8 Lambda表达式的函数式编程方式时实现命令设计模式。命令模式的目标是将请求封装成一个对象,从对客户端的不同类型请求,例如队列或日志请求参数化,并提供相应的操作。命令模式是一种通用编程方式,该方式基于运行时决策顺序来执行方法。模式的参与者如下:

  • 命令 :声明用于执行操作的接口。
  • 实体命令 :定义接收者对象和动作的绑定。
  • 客户端 :创建实体命令实例并设置它的接收者。
  • 调用者:控制命令来执行请求。
  • 接收者 :实际完成工作。

这些参与者之间的关系描述如下:

让我们看一个命令模式的具体例子,了解它是如何转换成lambda表达式的。假定我们有一个文件系统工具,所有动作都依赖它,例如打开文件,向文件写入和关闭文件。这可以实现宏功能 ,即一系列的操作可以被记录下来,之后作为一个单独操作执行。下面是我们的接收者。

public interface FileSystemReceiver {
void openFile();
void writeFile();
void closeFile();
}

每个操作都是命令,例如openFilewriteFile。我们可以创建一个通用命令接口来适配这些不同的操作。让我们将接口命名为Action,因为在我们的情境下,它代表执行一个操作。所有命令对象都需要实现这个接口。

public interface Action {
public void perform();
}

现在,让我们为每个操作实现Action接口。所有这些类需要做的就是调用FileReceiver的一个方法,并将这个调用封装到Action接口中。在封装操作后,使用适当的类命名规范来命名这些类。因此,openFile方法对象的类称为OpenFile

public class OpenFile implements Action {
private final FileReceiver fileReceiver;
public OpenFile(FileReceiver fileReceiver) {
this.fileReceiver = fileReceiver;
}
public void perform() {
fileReceiver.openFile();
}
}

现在,我们实现Macro类。每个宏包含一个动作序列,该序列可以依次执行动作,它将作为调用者。这个类可以记录动作,并一起运行。我们可以将动作序列存储在列表中,然后反复地获取每个动作来执行。

public class Macro {
private final List actions;
public Macro() {
actions = new ArrayList<>();
}
public void record(Action action) {
actions.add(action);
}
public void run() {
actions.forEach(Action::perform);
}
}

当填充宏对象时,我们可以添加每个命令的实例,这些实例也会被记录在宏对象里。现在简单的运行宏对象,每个命令将依次执行。我们的客户端代码如下。

Macro macro = new Macro();
macro.record(new OpenFile(fileReceiver));
macro.record(new WriteFile(fileReceiver));
macro.record(new CloseFile(fileReceiver));
macro.run();

如果你跟着我的思路读到这里,你会好奇lambda表达式如何适配这些的。实际上,所有命令类,例如OpenFile、WriteFile和CloseFile,其实只是lambda表达式想打破它们的封装。这些命令类只是在类之间传递。使用lambda表达式整个模式得到大大的简化,因为我们完全可以废除这些类。然我们看看宏类(客户端)使用lambda表达式代替命令类的效果。

Macro macro = new Macro();
macro.record(() -> fileReceiver.openFile());
macro.record(() -> fileReceiver.writeFile());
macro.record(() -> fileReceiver.closeFile());
macro.run();

如果能够意识到每个lambda表达式都在执行一个单独的方法调用,可以进一步改进。因此,可以直接使用方法引用。

Macro macro = new Macro();
macro.record(fileReceiver::openFile);
macro.record(fileReceiver::writeFile);
macro.record(fileReceiver::closeFile);
macro.run();

命令模式易于扩展,新的动作方法可以被添加到接收者中来创建新的命令实现,这样不需要改变任何客户端代码。JDK中的Runnable接口(java.lang.Runnable)是命令模式常用的流行接口。这篇博客中,我试着阐述带Java 8 lambda表达式的命令模式。你可以看到使用lambda表达式,少了很多样板代码,从而让代码变得更整洁。

时间: 2024-10-13 01:28:14

用Java 8 Lambda表达式实现设计模式:命令模式的相关文章

Java 8 Lambda表达式10个示例【存】

PS:不能完全参考文章的代码,请参考这个文件http://files.cnblogs.com/files/AIThink/Test01.zip 在Java 8之前,如果想将行为传入函数,仅有的选择就是匿名类,需要6行代码.而定义行为最重要的那行代码,却混在中间不够突出.Lambda表达式取代了匿名类,取消了模板,允许用函数式风格编写代码.这样有时可读性更好,表达更清晰.在Java生态系统中,函数式表达与对面向对象的全面支持是个激动人心的进步.将进一步促进并行第三方库的发展,充分利用多核CPU.尽

Java 8 lambda表达式20180404

Java 8 lambda表达式示例 我个人对Java 8发布非常激动,尤其是lambda表达式和流API.越来越多的了解它们,我能写出更干净的代码.虽然一开始并不是这样.第一次看到用lambda表达式写出来的Java代码时,我对这种神秘的语法感到非常失望,认为它们把Java搞得不可读,但我错了.花了一天时间做了一些lambda表达式和流API示例的练习后,我开心的看到了更清晰的Java代码.这有点像学习泛型,第一次见的时候我很讨厌它.我甚至继续使用老版Java 1.4来处理集合,直到有一天,朋

Java 8 lambda表达式示例

例1.用lambda表达式实现Runnable 我开始使用Java 8时,首先做的就是使用lambda表达式替换匿名类,而实现Runnable接口是匿名类的最好示例.看一下Java 8之前的runnable实现方法,需要4行代码,而使用lambda表达式只需要一行代码.我们在这里做了什么呢?那就是用() -> {}代码块替代了整个匿名类. 1 // Java 8之前: 2 new Thread(new Runnable() { 3 @Override 4 public void run() {

Java之Lambda表达式和Stream类简单例子

开心一笑 提出问题 java的lambda表达式和Stream如何使用??? 解决问题 Lambda表达式的语法 基本语法: (parameters) -> expression 或 (parameters) ->{ statements; } 看例子学习吧! 例一:定义一个AyPerson类,为之后的测试做准备. package com.evada.de; import java.util.Arrays; import java.util.List; class AyPerson{ priv

[Java 8] Lambda表达式对递归的优化(下) - 使用备忘录模式(Memoization Pattern)

使用备忘录模式(Memoization Pattern)提高性能 这个模式说白了,就是将需要进行大量计算的结果缓存起来,然后在下次需要的时候直接取得就好了.因此,底层只需要使用一个Map就够了. 但是需要注意的是,只有一组参数对应得到的是同一个值时,该模式才有用武之地. 在很多算法中,典型的比如分治法,动态规划(Dynamic Programming)等算法中,这个模式运用的十分广泛. 以动态规划来说,动态规划在求最优解的过程中,会将原有任务分解成若干个子任务,而这些子任务势必还会将自身分解成更

Java的Lambda表达式

Java的Lambda表达式 1. 什么是Lambda表达式 简单的说,Lambda表达式就是匿名方法.Lambda表达式让程序员能够使用更加简洁的代码,但是同样也使代码的可读性比较差. Lambda表达式也叫做匿名方法或者闭包. 2. 和匿名内部类做对比 Lambda是匿名方法,这个时候我们会想想到匿名内部类,我们来回想一下匿名内部类的用法,比如下面的代码就是使用匿名内部类实现了一个线程. public class Test { public static void main(String[]

设计模式 - 命令模式(command pattern) 具体解释

命令模式(command pattern) 详细解释 本文地址: http://blog.csdn.net/caroline_wendy 命令模式(command pattern) : 将请求封装成对象, 以便使用不同的请求\队列\日志来參数化其它对象. 命令模式也能够支持撤销操作. 简单的命令模式的实现: 1. 详细的类, 每个类都有特定的方法: /** * @time 2014年6月9日 */ package command; /** * @author C.L.Wang * */ publ

设计模式 - 命令模式(command pattern) 详解

命令模式(command pattern) 详解 本文地址: http://blog.csdn.net/caroline_wendy 命令模式: 将请求封装成对象, 以便使用不同的请求\队列\日志来参数化其他对象. 命令模式也支持可撤销操作. 命令模式: 调用者(Invoker); 命令(Command): 可执行方法(execute), 具体命令(Concrete Command); 接受者(Receiver): 调用命令(Set Command); 具体方法: 1. 具体对象. /** *

设计模式 - 命令模式(command pattern) 多命令 详解

命令模式(command pattern) 多命令 详解 本文地址: http://blog.csdn.net/caroline_wendy 参考命令模式: http://blog.csdn.net/caroline_wendy/article/details/31379977 具体步骤: 1. 多命令, 把未使用的命令, 初始化为空对象(NoCommand), 根据参数(slot), 选择输出命令. /** * @time 2014年6月16日 */ package command; /**