命令模式/command模式/行为型模式
举个栗子
指挥官向士兵下达命令,士兵执行
实现代码如下:
class Soldier {
public void exe() {
System.out.println("执行命令");
}
}
class Commander {
public void invok() {
Soldier soldier=new Soldier();
soldier.exe();
}
}
代码问题
上诉代码是最基本的一个实现,存在问题:
- Commander和Soldier高度耦合
- 新增命令的话,要改动Commander类,扩展性不好(影响已有实现),不符合OCP原则
- 指挥官指挥实现多条命令时,又要改Commander
- 士兵执行命令时如果加一些控制的话,如记录日志、延缓执行,加到哪都不合适,SRP和OCP原则。
重构后代码
class Soldier {
public void doCommandA() {
System.out.println("执行命令A");
}
public void doCommandB() {
System.out.println("执行命令B");
}
}
class CommandCenterA implements Command{
Soldier soldier;
public CommandCenterA(Soldier soldier) {
this.soldier=soldier;
}
@Override
public void exe() {
waitTime();
soldier.doCommandA();
soldier.doCommandB();
}
private void waitTime() {
System.out.println("延迟执行");
}
}
class Commander {
Command command;
public Commander(Command command) {
this.command=command;
}
public void exe(){
command.exe();
}
}
class Client{
public static void main(String[] args) {
Soldier soldier=new Soldier();
Command command=new CommandCenterA(soldier);//将soldier注入到命令中心
Commander commander=new Commander(command);//将命令中心注入到指挥官
commander.exe();
}
解决的问题
- Commander和Soldier隔离,解除耦合,
- 新增命令时,只需新增Command接口的实现CommandCenter,实现对命令和请求控制
- CommandCenter可以实现复合命令,延迟执行,记录日志等
命令模式
命令模式把一个请求或者操作封装到一个对象中,并把发出命令和执行命令隔离,使得发出命令不必关心命令如何执行。
讲个实例:
比如开关灯和开关空调。
最原始的,就是把两根电线连到一起,灯亮空调开,断开,灯灭空调关。(Receiver)
现在我们加了个开关,开关不仅能开灯,还能开空调。(将电灯开关和空调开关进一步抽象)
开关内部的实现,就是我们的命令中心。(开空调时可以设置温度,开灯时可以延缓执行)
将开关和电线隔离,并加入了我们对命令的具体控制。这就是命令模式。
Java实现要点 Invoker(请求者)+Commander(命令中心)+Receiver(执行者)
- 将请求者和执行者,加入中间层“命令中心”,将请求者对象绑定于一个动作
- 将命令中心抽象出接口,使得请求者依赖于接口而不是具体实现
- 命令中心实现对请求者和执行者的解耦
JDK中的命令模式
比较经典的应该就是Runnable了吧。
//执行者
Runnable soldier= new Runnable() {
public void run() {
System.out.println("执行命令");
}
};
//命令中心
Runnable thread=new Thread(solder);//可继承Thread继续定制
thread.sleep(1000);
//请求者
thread.start();
Thread源码:
/* What will be run. */
private Runnable target;
public Thread(Runnable target) {
init(null, target, "Thread-" + nextThreadNum(), 0);
}
/**
* If this thread was constructed using a separate
* <code>Runnable</code> run object, then that
* <code>Runnable</code> object's <code>run</code> method is called;
* otherwise, this method does nothing and returns.
* <p>
* Subclasses of <code>Thread</code> should override this method.
*
* @see #start()
* @see #stop()
* @see #Thread(ThreadGroup, Runnable, String)
*/
@Override
public void run() {
if (target != null) {
target.run();
}
}
注入了Runnable,并实现了对Runnable的控制。
优点
- 松耦合
- “命令中心”对命令可控(如日志化,延缓,复合命令),对请求可控(如队列,参数化)
- 扩展性。只需要实现新的“命令中心”即可,任意装配,并且不影响已有的实现,因为命令发起者(指挥官)中的命令中心时接口注入。
命令模式/command模式/行为型模式的更多相关文章
- NET设计模式 第二部分 创建型模式(6):创建型模式专题总结(Creational Pattern)
创建型模式专题总结(Creational Pattern) ——.NET设计模式系列之七 Terrylee,2006年1月 概述 创建型模式,就是用来创建对象的模式,抽象了实例化的过程.它帮助一个系统 ...
- 抽象工厂模式(abstract)创建型模式
(一)简单工厂模式? 现在的学习是面向对象面向接口的,但是执行时的操作需要实例化后的对象.随着我们需要的类的增加,我们就需要把这些共同的东西提取出来,放在一个抽象类中,让这些子类来继承抽象类.当我们调 ...
- 设计模式学习之命令模式(Command,行为型模式)(12)
一.命令模式的定义 命令模式属于对象的行为型模式.命令模式是把一个操作或者行为抽象为一个对象中,通过对命令的抽象化来使得发出命令的责任和执行命令的责任分隔开.命令模式的实现可以提供命令的撤销和恢复功能 ...
- 设计模式学习之建造者模式(Builder,创建型模式)(6)
假如我们需要建造一个房子,并且我们也不知道如何去建造房子,所以就去找别人帮我们造房子 第一步: 新建一个房子类House,里面有房子该有的属性,我们去找房子建造者接口HouseBuilder,我们要建 ...
- 设计模式学习之原型模式(Prototype,创建型模式)(5)
通过序列化的方式实现深拷贝 [Serializable] public class Person:ICloneable { public string Name { get; set; } publi ...
- ANDROID 中设计模式的采用--创建型模式
所谓模式就是在某一情景下解决某个问题的固定解决方案. 所有的创建型模式都是用作对象的创建或实例化的解决方案. 1 简单工厂模式 创建对象的最简单方法是使用new来创建一个对象,如果只创建一种固定 ...
- 设计模式学习之模板方法模式(TemplateMethod,行为型模式)(9)
一.什么是模板方法模式 Template Method模式也叫模板方法模式,是行为模式之一,它把具有特定步骤算法中的某些必要的处理委让给抽象方法,通过子类继承对抽象方法的不同实现改变整个算法的行为. ...
- 七、备忘录模式Memento(行为型模式)
其目的是,在不违反封装原则的前提下.採集和备份一个对象的内部状态以便这个对象能够在以后恢复到之前的某个状态. 在Memento模式中,有例如以下角色: 1.Memento (备忘录) * 存储Orig ...
- 设计模式-抽象工厂模式(AbstractFactory)(创建型模式)
//以下代码来源: 设计模式精解-GoF 23种设计模式解析附C++实现源码 //Product.h #pragma once class AbstractProductA { public: vir ...
- GoF的23种设计模式之行为型模式的特点和分类(2)
行为型模式用于描述程序在运行时复杂的流程控制,即描述多个类或对象之间怎样相互协作共同完成单个对象都无法单独完成的任务,它涉及算法与对象间职责的分配. 行为型模式分为类行为模式和对象行为模式,前者采用继 ...
随机推荐
- C#之数据分页
方法一:临时datatable 创建临时表,临时变量 DataTable dt = null; //临时表 ; //总分页数 ; //当前页数 ; //每页的数量 加载数据到临时表,该方法测试放到了窗 ...
- cf592d
题意:给出一个无根树,点数为10^5,所有边的长度为1.给定其中有一些点是受到攻击的. 现在要求一个人选定一个点作为起点,走遍所有的受攻击点(不用再回到起点). 需要的最短距离是多少,选定的起点是哪个 ...
- ASM:《X86汇编语言-从实模式到保护模式》第14章:保护模式下的特权保护和任务概述
★PART1:32位保护模式下任务的隔离和特权级保护 这一章是全书的重点之一,这一张必须要理解特权级(包括CPL,RPL和DPL的含义)是什么,调用门的使用,还有LDT和TSS的工作原理(15章着重 ...
- poj 2389.Bull Math 解题报告
题目链接:http://poj.org/problem?id=2389 题目意思:就是大整数乘法. 题目中说每个整数不超过 40 位,是错的!!!要开大点,这里我开到100. 其实大整数乘法还是第一次 ...
- plsql客户端显示菜单等
不小心把plsql的左边的都关了,如图 菜单条---工具---浏览器. 即可.
- 【XLL API 函数】xlStack
查看堆栈区还剩余多少空间 原型 Excel12(xlStack, LPXLOPER12 pxRes, 0); 参数 此函数没有带任何参数 属性值/返回值 返回堆栈区还剩余的字节数 备注 返回最新版本的 ...
- PHP安全编程:对输出要进行转义
为了区分数据是否已转义,我还是建议定义一个命名机制.对于输出到客户机的转义数据,我使用$html数组进行存储,该数据首先初始化成一个空数组,对所有已过滤和已转义数据进行保存. 1 <?php 2 ...
- [Android Pro] Android打包一个Apk后,如何查看它的VersionCode、VersionName 等等。
Android打包成Apk后,其实是一个压缩文件,我们用winrar打开也能看到里面的文件结构.还能看到AndroidManifest.但是里面的内容有点问题. 不知道是因为加密还是Android就是 ...
- 在HTML中禁止文字的复制
很简单,只需在<body>中添加如下代码: <body oncontextmenu='return false' ondragstart='return false' onsele ...
- XMPP框架下微信项目总结(2)授权登陆/注销/注册/打印日志
xmpp授权登陆步骤1 初始化xmppstream 连接服务器 传递属性jid(IP地址 端口号)2 连接成功后 传递“登”陆密码授权 3 授权后,发送在线消息xmpp所有的代理都是子线程中调用的,处 ...