阿里P9大佬分享:如何让代码更加灵活

简介: 阿里P9大佬分享:如何让代码更加灵活

面试官: 你好,今天我们要讨论的是命令模式。首先,你能解释一下什么是命令模式吗?

求职者: 当然可以。命令模式是一种行为设计模式,它将一个请求封装成一个对象,从而让你使用不同的请求、队列或者日志请求来参数化其他对象。命令模式也支持可撤销的操作。

面试官: 很好。那么,你能给我举一个命令模式的实际编程例子吗?

求职者: 当然可以。就像遥控器一样,它有很多按钮,每个按钮背后都有一个命令。在编程中,我们可以创建一个命令接口,这个接口定义了执行操作的方法。然后,我们可以为每个操作创建具体的命令类。这些类包含了执行操作所必需的信息和方法。

例如,在一个游戏模拟器中,我们可以有启动和停止游戏的命令。启动命令会调用游戏控制台的start_game方法,而停止命令会调用stop_game方法。

面试官: 那么,命令模式如何实现撤销和恢复操作呢?

求职者: 要实现撤销和恢复操作,命令对象需要存储原始状态信息,以便可以恢复到执行命令之前的状态。通常,每个命令类会有一个undo方法来回滚操作。在执行操作时,命令对象被添加到历史记录中。当执行撤销操作时,可以从历史记录中获取最新的命令并调用其undo方法。

面试官: 很好。那么,你认为命令模式的优点是什么?

求职者: 命令模式的优点包括:

  1. 解耦发送者和接收者:发送命令的对象不需要知道接收者是谁,也不需要知道被执行的操作的具体细节。
  2. 扩展性:可以很容易地添加新的命令,因为命令模式使用命令作为中间层。
  3. 组合命令:可以组合多个命令来实现宏命令。
  1. 支持撤销和恢复:命令模式可以通过实现undo方法来支持撤销和恢复操作。

面试官: 非常好。你提到了命令模式的关键优点。那么,命令模式是如何将创建型模式和行为型模式交织在一起的呢?

求职者: 命令模式可以看作是创建型模式和行为型模式的结合。从创建型模式的角度来看,命令模式涉及到创建具体命令对象,并与特定的接收者关联。而从行为型模式的角度来看,命令模式定义了一个执行操作的接口,使得发送者和接收者之间的请求调用可以解耦。

面试官: 很好,现在让我们来探讨一下命令模式如何支持宏命令的组合。首先,你能解释一下什么是宏命令吗?

求职者: 当然可以。宏命令是一组命令的集合,它可以一起被执行。在命令模式中,我们可以创建一个宏命令对象,这个对象内部包含了一组命令对象。当执行宏命令时,它会依次执行内部的每个命令。

面试官: 那你能用代码示例来说明如何实现宏命令吗?

求职者: 当然。我们可以创建一个MacroCommand类,它实现了Command接口。这个类内部维护了一个命令列表,并提供了添加命令的方法。在execute方法中,它会遍历并执行所有的命令。

class MacroCommand implements Command {
    private List<Command> commands;

    public MacroCommand() {
        commands = new ArrayList<>();
    }

    public void addCommand(Command command) {
        commands.add(command);
    }

    public void execute() {
        for (Command command : commands) {
            command.execute();
        }
    }

    public void undo() {
        // Optional: Implement undo in reverse order if needed
        ListIterator<Command> iterator = commands.listIterator(commands.size());
        while (iterator.hasPrevious()) {
            iterator.previous().undo();
        }
    }
}

在这个例子中,MacroCommand可以包含任何数量的命令对象。当调用它的execute方法时,它会执行所有添加的命令。我们还可以实现undo方法,以便可以撤销宏命令中的所有操作。

面试官: 现在,我们来看一个具体的例子,说明如何在命令模式中实现命令的撤销功能。你能给我一个简单的代码示例吗?

求职者: 当然可以。假设我们有一个简单的文本编辑器,我们可以添加文本和删除文本。我们将创建一个添加文本的命令,并实现一个撤销功能,这样我们就可以撤销添加的文本

首先,我们定义一个Command接口,它包含executeundo方法:

interface Command {
    void execute();
    void undo();
}

然后,我们创建一个AddTextCommand类,它实现了Command接口

class AddTextCommand implements Command {
    private String textToAdd;
    private TextEditor editor;

    public AddTextCommand(String textToAdd, TextEditor editor) {
        this.textToAdd = textToAdd;
        this.editor = editor;
    }

    public void execute() {
        editor.addText(textToAdd);
    }

    public void undo() {
        editor.removeText(textToAdd);
    }
}

接下来,我们定义TextEditor类,它包含添加和删除文本的功能:

class TextEditor {
    private StringBuilder text = new StringBuilder();

    public void addText(String textToAdd) {
        text.append(textToAdd);
    }

    public void removeText(String textToRemove) {
        int index = text.lastIndexOf(textToRemove);
        if (index != -1) {
            text.delete(index, index + textToRemove.length());
        }
    }

    public String getText() {
        return text.toString();
    }
}

现在,我们需要一个机制来执行和撤销命令,我们创建一个TextEditorDriver类:

class TextEditorDriver {
    private Stack<Command> commandHistory = new Stack<>();

    public void executeCommand(Command cmd) {
        cmd.execute();
        commandHistory.push(cmd);
    }

    public void undoLastCommand() {
        if (!commandHistory.isEmpty()) {
            Command lastCmd = commandHistory.pop();
            lastCmd.undo();
        }
    }
}

最后,我们在main方法中模拟命令的执行和撤销:

public class Main {
    public static void main(String[] args) {
        TextEditor editor = new TextEditor();
        TextEditorDriver driver = new TextEditorDriver();

        Command addTextCmd = new AddTextCommand("Hello, World!", editor);
        driver.executeCommand(addTextCmd);

        System.out.println("Text after add command: " + editor.getText());

        driver.undoLastCommand();

        System.out.println("Text after undo: " + editor.getText());
    }
}

在这个例子中,我们添加了文本"Hello, World!"到文本编辑器,然后我们执行了撤销操作,这将移除我们刚刚添加的文本。

面试官: 好的,让我们离开文本编辑器,看看命令模式中撤销功能在其他场景中的应用。你能给我一个不同的例子吗?

求职者: 当然。让我们考虑一个家居自动化系统,比如智能灯泡的控制。我们可以开灯和关灯,并且我们希望能够撤销这些操作。

首先,我们定义一个Command接口:

interface Command {
    void execute();
    void undo();
}

接着,我们创建Light类和两个命令:LightOnCommandLightOffCommand

class Light {
    private boolean isOn = false;

    public void toggleLight() {
        isOn = !isOn;
        if (isOn) {
            System.out.println("Light is ON");
        } else {
            System.out.println("Light is OFF");
        }
    }
}

class LightOnCommand implements Command {
    private Light light;

    public LightOnCommand(Light light) {
        this.light = light;
    }

    public void execute() {
        light.toggleLight();
    }

    public void undo() {
        light.toggleLight();
    }
}

class LightOffCommand implements Command {
    private Light light;

    public LightOffCommand(Light light) {
        this.light = light;
    }

    public void execute() {
        light.toggleLight();
    }

    public void undo() {
        light.toggleLight();
    }
}

然后,我们创建一个控制器,它可以执行命令并且支持撤销:

class RemoteControl {
    private Command lastCommand;

    public void submit(Command command) {
        command.execute();
        lastCommand = command;
    }

    public void undoLastCommand() {
        if (lastCommand != null) {
            lastCommand.undo();
        }
    }
}

最后,我们在main方法中模拟命令的执行和撤销

public class SmartHomeDemo {
    public static void main(String[] args) {
        Light livingRoomLight = new Light();
        Command lightOn = new LightOnCommand(livingRoomLight);
        Command lightOff = new LightOffCommand(livingRoomLight);

        RemoteControl remoteControl = new RemoteControl();
        remoteControl.submit(lightOn);  // Light is ON
        remoteControl.submit(lightOff); // Light is OFF

        // Oops, didn't mean to turn it off. Let's undo that.
        remoteControl.undoLastCommand(); // Light is ON
    }
}

在这个例子中,我们通过RemoteControl执行了开灯和关灯的命令,并且使用撤销功能来撤销关灯的操作。

面试官: 很好,这个例子清楚地展示了命令模式在家居自动化系统中的撤销功能。你展示了命令模式的灵活性和命令撤销的实用性。这就是我们今天要讨论的全部内容,谢谢你的参与。


相关文章
|
弹性计算 分布式计算 监控
ECS飘移问题之飘移无法实现如何解决
ECS(Elastic Compute Service,弹性计算服务)是云计算服务提供商提供的一种基础云服务,允许用户在云端获取和配置虚拟服务器。以下是ECS服务使用中的一些常见问题及其解答的合集:
|
10月前
|
机器学习/深度学习 自然语言处理 并行计算
探索深度学习与自然语言处理的最新进展
探索深度学习与自然语言处理的最新进展
290 4
|
9月前
|
机器学习/深度学习 人工智能 搜索推荐
AI技术在医疗领域的应用与前景####
本文深入探讨了人工智能(AI)技术在医疗健康领域中的多维度应用,从疾病诊断、个性化治疗到健康管理,展现了AI如何革新传统医疗模式。通过分析当前实践案例与最新研究成果,文章揭示了AI技术提升医疗服务效率、精准度及患者体验的巨大潜力,并展望了其在未来医疗体系中不可或缺的地位。 ####
|
运维 监控 API
自动化运维实践指南:Python脚本优化服务器管理任务
本文探讨了Python在自动化运维中的应用,介绍了使用Python脚本优化服务器管理的四个关键步骤:1) 安装必备库如paramiko、psutil和requests;2) 使用paramiko进行远程命令执行;3) 利用psutil监控系统资源;4) 结合requests自动化软件部署。这些示例展示了Python如何提升运维效率和系统稳定性。
860 8
|
存储 安全 数据库
云计算:架构、类型及其优缺点
【8月更文挑战第20天】
2826 0
|
SQL XML 安全
Mybatis分页方式详解
Mybatis分页方式详解
1514 0
|
存储 监控 API
【云原生系列】云计算概念与架构设计介绍
云计算是一种基于互联网的计算模式,在这个模式下,各种计算资源(例如计算机、存储设备、网络设备、应用程序等)可以通过互联网实现共享和交付。云计算架构设计的主要目标是实现高效、可扩展、可靠、安全和经济的计算资源共享。
531 5
|
存储 监控 NoSQL
干货文:企业 IT 基础架构|(精华篇)
干货文:企业 IT 基础架构|(精华篇)
1239 1
|
C#
C#高性能二进制序列化
二进制序列化可以方便快捷的将对象进行持久化或者网络传输,并且体积小、性能高,应用面甚至还要高于json的序列化;开始之前,先来看看dotcore/dotne自带的二进制序列化:C#中对象序列化和反序列化一般是通过BinaryFormatter类来实现的二进制序列化、反序列化的。
2103 0
|
开发者
泡泡龙游戏开发实战教程(8):完整流程
泡泡龙游戏开发实战教程(8):完整流程
358 0