当前位置: 首页 > news >正文

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

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

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

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

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

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

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

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

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

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

  1. 解耦发送者和接收者:发送命令的对象不需要知道接收者是谁,也不需要知道被执行的操作的具体细节。
  2. 扩展性:可以很容易地添加新的命令,因为命令模式使用命令作为中间层。
  3. 组合命令:可以组合多个命令来实现宏命令。
  4. 支持撤销和恢复:命令模式可以通过实现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 neededListIterator<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 ONremoteControl.submit(lightOff); // Light is OFF// Oops, didn't mean to turn it off. Let's undo that.remoteControl.undoLastCommand(); // Light is ON}
}

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

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

相关文章:

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

面试官: 你好&#xff0c;今天我们要讨论的是命令模式。首先&#xff0c;你能解释一下什么是命令模式吗&#xff1f; 求职者: 当然可以。命令模式是一种行为设计模式&#xff0c;它将一个请求封装成一个对象&#xff0c;从而让你使用不同的请求、队列或者日志请求来参数化其他…...

SpringBoot中加载配置文件的优先级

在Spring Boot中&#xff0c;加载配置文件的优先级按照以下顺序进行&#xff0c;后续的配置会覆盖之前的配置&#xff1a; 默认属性&#xff1a;这些属性在Spring Boot本身中定义&#xff0c;并且通常是不可变的。它们作为后备值。 应用程序属性&#xff1a;这些属性在应用程序…...

Mysql命令行客户端

命令行客户端 操作数据库操作数据表 操作数据库 mysql> create database mike charsetutf8; Query OK, 1 row affected (0.01 sec) mysql> show databases; -------------------- | Database | -------------------- | information_schema | | mike …...

开源的python 游戏开发库介绍

本文将为您详细讲解开源的 Python 游戏开发库&#xff0c;以及它们的特点、区别和应用场景。Python 社区提供了多种游戏开发库&#xff0c;这些库可以帮助您在 Python 应用程序中实现游戏逻辑、图形渲染、声音处理等功能。 1. Pygame 特点 - 基于 Python 的游戏开发库。…...

批量提取PDF指定区域内容到 Excel 以及根据PDF里面第一页的标题来批量重命名-附思路和代码实现

首先说明下&#xff0c;PDF需要是电子版本的&#xff0c;不能是图片或者无法选中的那种。 需求1&#xff1a;假如我有一批数量比较多的同样格式的PDF电子文档&#xff0c;需要把特定多个区域的数字或者文字提取出来 需求2&#xff1a;我有一批PDF文档&#xff0c;但是文件的名…...

PCM会重塑汽车OTA格局吗(1)

目录 1.汽车OTA概述 2.ST如何考虑OTA&#xff1f; 2.1 Stellar四大亮点 2.2 PCM技术视角下的OTA 3.小结 1.汽车OTA概述 随着智能网联汽车的飞速发展&#xff0c;汽车OTA也越来越盛行&#xff1b; 目前来讲OTA分为FOTA和SOTA(Software-over-the-air)两种&#xff0c;区别…...

Intel® Extension for PyTorch*详细安装教程

最近在研究Intel的pytorch的加速拓展Intel Extension for PyTorch*,但是发现官网的文档全是英文的&#xff0c;不太好找安装教程。所以特此分享Intel Extension for PyTorch*的详细安装教程。 文章目录 一、安装所需系统要求1.1 硬件需求1.2 软件需求 二、准备2.1 安装驱动程序…...

云上攻防-云产品篇堡垒机场景JumpServer绿盟SASTeleport麒麟齐治

知识点 1、云产品-堡垒机-产品介绍&攻击事件 2、云产品-堡垒机-安全漏洞&影响产品 章节点&#xff1a; 云场景攻防&#xff1a;公有云&#xff0c;私有云&#xff0c;混合云&#xff0c;虚拟化集群&#xff0c;云桌面等 云厂商攻防&#xff1a;阿里云&#xff0c;腾讯…...

【顶刊|修正】多区域综合能源系统热网建模及系统运行优化【复现+延伸】

目录 主要内容 部分代码 结果一览 下载链接 主要内容 该程序复现《多区域综合能源系统热网建模及系统运行优化》模型并进一步延伸&#xff0c;基于传热学的基本原理建立了区域热网能量传输通用模型&#xff0c;对热网热损方程线性化实现热网能量流建模&#xff0…...

使用Numpy手工模拟梯度下降算法

代码 import numpy as np # Compute every step manually# Linear regression # f w * x # here : f 2 * x X np.array([1, 2, 3, 4], dtypenp.float32) Y np.array([2, 4, 6, 8], dtypenp.float32)w 0.0# model output def forward(x):return w * x# loss MSE def loss…...

金融数据采集与风险管理:Open-Spider工具的应用与实践

一、项目介绍 在当今快速发展的金融行业中&#xff0c;新的金融产品和服务层出不穷&#xff0c;为银行业务带来了巨大的机遇和挑战。为了帮助银行员工更好地应对这些挑战&#xff0c;我们曾成功实施了一个创新的项目&#xff0c;该项目采用了先进的爬虫技术&#xff0c;通过ope…...

鸿蒙Harmony应用开发—ArkTS声明式开发(通用属性:动态属性设置)

动态设置组件的属性&#xff0c;支持开发者在属性设置时使用if/else语法&#xff0c;且根据需要使用多态样式设置属性。 说明&#xff1a; 从API Version 11开始支持。后续版本如有新增内容&#xff0c;则采用上角标单独标记该内容的起始版本。 attributeModifier attributeMo…...

Vue class和style绑定:动态美化你的组件

&#x1f90d; 前端开发工程师、技术日更博主、已过CET6 &#x1f368; 阿珊和她的猫_CSDN博客专家、23年度博客之星前端领域TOP1 &#x1f560; 牛客高级专题作者、打造专栏《前端面试必备》 、《2024面试高频手撕题》 &#x1f35a; 蓝桥云课签约作者、上架课程《Vue.js 和 E…...

[C++] Windows中字符串函数的种类

文章目录 C标准库函数VC CRT函数Win32 APILinux C标准库函数 #include || #include <string.h> || #include 都可以使用以下函数&#xff1a; char *strcpy(char *dest, const char *src) //将Src字符串拷贝到Dst字符串地址。没有目标内存大小检查&#xff0c;可能会导致…...

Django工具

一、分页器介绍 1.1、介绍 分页,就是当我们在页面中显示一些信息列表,内容过多,一个页面显示不完,需要分成多个页面进行显示时,使用的技术就是分页技术 在django项目中,一般是使用3种分页的技术: 自定义分页功能,所有的分页功能都是自己实现django的插件 django-pagin…...

vue ui Starting GUI 图形化配置web新项目

前言&#xff1a;在vue框架里面&#xff0c; 以往大家都是习惯用命令行 vue create 、vue init webpack创建新前端项目&#xff0c;而vue ui是一个可视化的图形界面&#xff0c;对于新手来说更加友好了&#xff0c;不但可以创建、管理、还可以更新vue项目&#xff0c;也可以下载…...

Unity InputField宽度自适应内容

在Unity中&#xff0c;InputField在我们输入内容时&#xff0c;只会显示适应初始宽度的最新内容&#xff0c;或者自定义长度内容。 那么&#xff0c;要实现宽度自适应内容就需要另寻他法了。 以下是通过一个控制脚本来实现的一个简单方法。 直接上脚本&#xff1a; using S…...

加快代码审查的 7 个最佳实践

目录 前言 1-保持小的拉取请求 2-使用拉取请求模板 3-实施响应时间 SLA 4-培训初级和中级工程师 5-设置持续集成管道 6-使用拉取请求审查应用程序 7-生成图表以可视化您的代码更改 前言 代码审查可能会很痛苦软件工程师经常抱怨审查过程缓慢&#xff0c;延迟下游任务&…...

C++读写Excel(xlnt库的使用)

一、简介 官网&#xff1a;https://github.com/tfussell/xlnt Cross-platform user-friendly xlsx library for C11 xlnt is a modern C library for manipulating spreadsheets in memory and reading/writing them from/to XLSX files as described in ECMA 376 4th edition…...

【工具】conda常用命令

Conda 是一个流行的包管理器和环境管理器&#xff0c;用于安装、部署和管理软件包及其依赖项。 创建环境&#xff1a; conda create --name myenv 这将创建一个名为 myenv 的新环境。 激活环境&#xff1a; conda activate myenv 这会激活名为 myenv 的环境。在 Windows 上&am…...

后进先出(LIFO)详解

LIFO 是 Last In, First Out 的缩写&#xff0c;中文译为后进先出。这是一种数据结构的工作原则&#xff0c;类似于一摞盘子或一叠书本&#xff1a; 最后放进去的元素最先出来 -想象往筒状容器里放盘子&#xff1a; &#xff08;1&#xff09;你放进的最后一个盘子&#xff08…...

IDEA运行Tomcat出现乱码问题解决汇总

最近正值期末周&#xff0c;有很多同学在写期末Java web作业时&#xff0c;运行tomcat出现乱码问题&#xff0c;经过多次解决与研究&#xff0c;我做了如下整理&#xff1a; 原因&#xff1a; IDEA本身编码与tomcat的编码与Windows编码不同导致&#xff0c;Windows 系统控制台…...

【Axure高保真原型】引导弹窗

今天和大家中分享引导弹窗的原型模板&#xff0c;载入页面后&#xff0c;会显示引导弹窗&#xff0c;适用于引导用户使用页面&#xff0c;点击完成后&#xff0c;会显示下一个引导弹窗&#xff0c;直至最后一个引导弹窗完成后进入首页。具体效果可以点击下方视频观看或打开下方…...

生成xcframework

打包 XCFramework 的方法 XCFramework 是苹果推出的一种多平台二进制分发格式&#xff0c;可以包含多个架构和平台的代码。打包 XCFramework 通常用于分发库或框架。 使用 Xcode 命令行工具打包 通过 xcodebuild 命令可以打包 XCFramework。确保项目已经配置好需要支持的平台…...

MFC内存泄露

1、泄露代码示例 void X::SetApplicationBtn() {CMFCRibbonApplicationButton* pBtn GetApplicationButton();// 获取 Ribbon Bar 指针// 创建自定义按钮CCustomRibbonAppButton* pCustomButton new CCustomRibbonAppButton();pCustomButton->SetImage(IDB_BITMAP_Jdp26)…...

Linux相关概念和易错知识点(42)(TCP的连接管理、可靠性、面临复杂网络的处理)

目录 1.TCP的连接管理机制&#xff08;1&#xff09;三次握手①握手过程②对握手过程的理解 &#xff08;2&#xff09;四次挥手&#xff08;3&#xff09;握手和挥手的触发&#xff08;4&#xff09;状态切换①挥手过程中状态的切换②握手过程中状态的切换 2.TCP的可靠性&…...

dedecms 织梦自定义表单留言增加ajax验证码功能

增加ajax功能模块&#xff0c;用户不点击提交按钮&#xff0c;只要输入框失去焦点&#xff0c;就会提前提示验证码是否正确。 一&#xff0c;模板上增加验证码 <input name"vdcode"id"vdcode" placeholder"请输入验证码" type"text&quo…...

ServerTrust 并非唯一

NSURLAuthenticationMethodServerTrust 只是 authenticationMethod 的冰山一角 要理解 NSURLAuthenticationMethodServerTrust, 首先要明白它只是 authenticationMethod 的选项之一, 并非唯一 1 先厘清概念 点说明authenticationMethodURLAuthenticationChallenge.protectionS…...

JVM暂停(Stop-The-World,STW)的原因分类及对应排查方案

JVM暂停(Stop-The-World,STW)的完整原因分类及对应排查方案,结合JVM运行机制和常见故障场景整理而成: 一、GC相关暂停​​ 1. ​​安全点(Safepoint)阻塞​​ ​​现象​​:JVM暂停但无GC日志,日志显示No GCs detected。​​原因​​:JVM等待所有线程进入安全点(如…...

什么是Ansible Jinja2

理解 Ansible Jinja2 模板 Ansible 是一款功能强大的开源自动化工具&#xff0c;可让您无缝地管理和配置系统。Ansible 的一大亮点是它使用 Jinja2 模板&#xff0c;允许您根据变量数据动态生成文件、配置设置和脚本。本文将向您介绍 Ansible 中的 Jinja2 模板&#xff0c;并通…...