大家好,欢迎来到设计模式专题,我们的主旨是介绍一些有趣好玩的设计模式。

今天我们介绍的设计模式叫做命令模式(command),在这个模式下,我们可以实现do和undo的解耦,让使用方不用关心内部的实现细节。

command模式

这个模式我们在日常当中经常使用,举一个很简单的例子,比如说我们发布代码。发布了之后发现不小心发布上去了一个bug,这个时候我们应该做什么?很简单,就是回滚,把线上的代码回滚到这一次发布之前的代码。这样我们这次发布带来的改动就会被消除,那么就避免了bug的产生。

那么,对于一个发布系统来说,它需要做什么?其实也就是两个功能,一个是发布另外一个是回滚。这两个操作是互相可逆的,对于它的使用者来说,是不会关心它的内部是如何实现的,我们只需要在页面上按按钮就好了。

我们来回顾一下这个过程,我们点击发布,可以把最新的代码发布上线。发布之后发现问题,再点击回滚,系统再自动恢复到发布之前的状态。发布和回滚彼此是可逆的,当我们消除掉bug之后,再次点击发布,又可以再次发布最新的代码了。

command模式就是做的这个事情,也就是对do和undo的封装。我们来看一个很简单的例子,对文件改名。比如说我们要把系统当中的文件改名,从A.txt改成B.txt。这个功能很简单,系统为我们提供了现成的函数,叫做os.rename(),我们只需要把A和B两个文件的地址传入其中即可。

假如我们发现改名字改错了,想回滚怎么办呢?会发现我们改动之前的名字已经忘了,不知道怎么回滚了。这个时候就可以使用command模式,我们来看代码:

import os

class MoveFileCommand:

    def __init__(self, src, dest):
        self.src = src
        self.dest = dest

    def execute(self):
        self.rename(self.src, self.dest)

    def undo(self):
        self.rename(self.dest, self.src)

    def rename(self, src, dest):
        print('renaming from {} to {}'.format(src, dest))
        os.rename(src, dest)

在execute方法当中,我们把文件从src变成了dest,如果想要回滚,它又会再次调用rename。将文件名从dest回滚到src。这样的话,作为使用方就可以完全不用理解api内部的实现逻辑了,不然的话为了防止改错了的情况,还需要做很多适配。

menu item

有了command模式之后我们可以在外面在封装一层用来ui交互上,我们很常见的一种UI交互方式就是按钮。某一个按钮点一下之后会出现一个按过的标记,并且实现一个什么功能。再按一次标记消失,功能也随之关闭。

我随便找了一个例子,比如下图菜单当中的show minimap,show breadcrumbs这些都是这样的功能。点一下出现缩略图,再点一下缩略图消失。

如果你写过UI页面的话,一般来说我们会先定义一个Menu Item的类,表示菜单当中的所有的item的基类。不同的选项表示不同的item,我们进一步分析会发现有些item我们需要这样双击关闭的机制,而有些item是没有的。比如上面的Run、Output这些item都是点一次执行一次的。

我们当然可以把上面介绍的Command对象直接当做item,但是这样不利于整个菜单的统一,所以我们还会在外面包一层。比如所有MenuItem的父类应该是这样的:

class MenuItemBaseClass:
    def __init__(self):
        pass
    
    def pressed(self):
        pass
    
    def unpress(self):
        pass

有了这个基类之后,我们就可以实现一个可回滚的类,将command的对象作为类成员变量,再在其中实现unpress方法:

class RedoableMenu(MenuItemBaseClass):
    def __init__(self, command):
        self_command = command
        
    def pressed(self):
        self._command.execute()
        
    def unpress(self):
        self._command.undo()

这样我们的UI就和command解耦了,如果我们想要实现不同的可以回滚的功能, 只需要实现不同的command创建实例就可以了。对于整个UI的使用没有任何影响,UI组件当中用到的所有类都是统一的。可能在Python这种弱类型语言当中看不太出来,因为我们一个list说是menu基类的list,但是其实装什么都行。但如果是强类型语言,那么这种抽象和封装就是非常有必要的了。

今天的文章就到这里,衷心祝愿大家每天都有所收获。如果还喜欢今天的内容的话,请来一个三连支持吧~(点赞、关注、转发

原文链接,求个关注

- END -

{{uploading-image-118988.png(uploading...)}}

详解command设计模式,解耦操作和回滚的更多相关文章

  1. MySQL【Update误操作】回滚(转)

    前言:      继上一篇MySQL[Delete误操作]回滚之后,现在介绍下Update回滚,操作数据库时候难免会因为“大意”而误操作,需要快速恢复的话通过备份来恢复是不太可能的,因为需要还原和bi ...

  2. java SSM多操作注解回滚

    在业务操作时难免会遇到一个业务多操作,会用到事物回滚这里写了一个简单的多操作失败事物回滚案例 在这之前你需要在你的applicationContext-mybatis.xml中配置: <!-- ...

  3. java开发中的23中设计模式详解--大话设计模式

    设计模式(Design Patterns) ——可复用面向对象软件的基础 设计模式(Design pattern)是一套被反复使用.多数人知晓的.经过分类编目的.代码设计经验的总结.使用设计模式是为了 ...

  4. RDD的详解、创建及其操作

    RDD的详解 RDD:弹性分布式数据集,是Spark中最基本的数据抽象,用来表示分布式集合,支持分布式操作! RDD的创建 RDD中的数据可以来源于2个地方:本地集合或外部数据源 RDD操作 分类 转 ...

  5. 算法进阶面试题01——KMP算法详解、输出含两次原子串的最短串、判断T1是否包含T2子树、Manacher算法详解、使字符串成为最短回文串

    1.KMP算法详解与应用 子序列:可以连续可以不连续. 子数组/串:要连续 暴力方法:逐个位置比对. KMP:让前面的,指导后面. 概念建设: d的最长前缀与最长后缀的匹配长度为3.(前缀不能到最后一 ...

  6. EditPlus正则表达式中英文详解(附常用事例操作)

    http://www.cnblogs.com/JustinYoung/articles/editplus_regular_expression.html EditPlus正则表达式中英文详解 \t T ...

  7. pandas常用操作详解——pandas的去重操作df.duplicated()与df.drop_duplicates()

    df.duplicated() 参数详解: subset:检测重复的数据范围.默认为数据集的所有列,可指定特定数据列: keep: 标记哪个重复数据,默认为'first'.1.'first':标记重复 ...

  8. 详解java设计模式之责任链模式

    详见:http://blog.yemou.net/article/query/info/tytfjhfascvhzxcyt175 从击鼓传花谈起 击鼓传花是一种热闹而又紧张的饮酒游戏.在酒宴上宾客依次 ...

  9. 详解JS设计模式

    原文链接:www.cnblogs.com 一:理解工厂模式 工厂模式类似于现实生活中的工厂可以产生大量相似的商品,去做同样的事情,实现同样的效果;这时候需要使用工厂模式. 简单的工厂模式可以理解为解决 ...

随机推荐

  1. LAMP 和 LNMP

    #0x01 组成: LAMP==Linux+Apache+Mysql+PHP LNMP==Linux+Nginx+Mysql+PHP LANMP==linux + nginx + apache + m ...

  2. get_started_3dsctf_2016

    题外:这道题不是很难,但是却难住了我很久.主要是在IDA中查看反编译出的伪代码时双击了一下gets()函数,结果进入gets函数内部,我当时就懵了,误以为这是一个自定义函数,但是自定义函数应该应该不能 ...

  3. PDF启动增加字段

    ALTER TABLE `cnoa_system_fs` ADD `reviewStatus` INT(1) NOT NULL DEFAULT '0' AFTER `isArchive`;ALTER  ...

  4. Linux:网络基础配置

    一.修改主机名 hostname 查看主机名 1.hostname   zy 修改主机名为zy,临时生效,重新登录系统生效. 2.想要永久修改,,需修改配置文件:  vi   /etc/sysconf ...

  5. 自然常数e的含义

    e是一个重要的常数,但是它的直观含义却不像 π 那么明了.我们都知道,圆的周长与直径之比是一个常数,这个常数被称为圆周率,记作 π = 3.14159......可是e代表什么呢? e是“指数”(ex ...

  6. 关于java基础_数组的学习

    数组的学习 1.数组的概念?作用是什么? 系统中存储多个值, 2.数组的定义? 数据类型[] 数组名; 3.定义好数组以后需要对其进行初始化 数组初始化有两种: 第一种动态初始化,指定数组的长度,长度 ...

  7. 数据库SQL调优的几种方式

    1.创建索引 (1) 要尽量避免全表扫描,首先应考虑在where 及order by涉及的列上建立索引 (2) 在经常需要进行检索的字段上创建索引,一个表中的索引最好不要超过6个 2.避免在索引上使用 ...

  8. 有向图的基本算法-Java实现

    有向图 有向图同无向图的区别为每条边带有方向,表明从一个顶点至另一个顶点可达.有向图的算法多依赖深度搜索算法. 本文主要介绍有向图的基本算法,涉及图的表示.可达性.检测环.图的遍历.拓扑排序以及强连通 ...

  9. python-文本操作和二进制储存

    0x01 open方法 r read w write a append b byte test.txt内容为 yicunyiye wutang 读取test.txt f = open('test.tx ...

  10. 【论文】The Road to SDN: An Intellectual History of Programmable Networks

    目录 ABSTRACT: 1 Introduction: 2 The Road to SDN: 2.1 Active Networking Technology push and use pull I ...