3.3设计模式之命令模式(Command)

3.3.1 模式意图:

在开发中会遇到多个技能排序释放;事件到达一定阈值时触发;对操作进行"记录、撤销、重做"等行为。在以上情况下,需要将"行为请求者"与"行为实现者"解耦。将一组行为抽象为对象,可以实现二者之间的松耦合。为了满足这种需求,可以考虑使用命令模式

3.3.2 模式概念:

它属于行为型模式,以对象的形式将请求封装,从而让你使用不同的请求把客户端参数化。对排队的请求或记录请求相关日志,可以提供命令的恢复、撤销功能。

3.3.3 模式元素:

  • 命令类抽象(Command)
  • 命令类细节(CommandOne、CommandTwo)
  • 调用者(Invoker):
  • 接收者(Receiver)

3.3.4 代码示例:

接受者

public interface IReceiver
{
    void Release();
}
public class SkillOne : IReceiver
{
    public void Release()
    {
        Debug.Log("释放技能一");
    }
}
public class SkillTwo : IReceiver
{
    public void Release()
    {
        Debug.Log("释放技能二");
    }
}

命令相关

public abstract class Command
{
    protected IReceiver receiver = null;

    public Command(IReceiver receiver)
    {
        this.receiver = receiver;
    }
    public abstract void Execute();
}

public class CommandOne : Command
{
    public CommandOne(IReceiver receiver) : base(receiver) { }

    public override void Execute()
    {
        Debug.Log("命令一");
        receiver.Release();
    }
}

public class CommandTwo : Command
{
    public CommandTwo(IReceiver receiver) : base(receiver) { }

    public override void Execute()
    {
        Debug.Log("命令二");
        receiver.Release();
    }
}

调用者

public class Invoker
{
    public List<Command> commadGroup = new List<Command>();

    public void Execute()
    {
        foreach (var commad in commadGroup)
        {
            commad.Execute();
        }
    }

    public void Add(Command command)
    {
        commadGroup.Add(command);
    }
}

调用示例

    void Start()
    {
        Invoker invoker = new Invoker();
        invoker.Add(new CommandOne(new SkillOne()));
        invoker.Add(new CommandTwo(new SkillTwo()));

        invoker.Execute();
    }

3.3.5 写法对比:

3.3.6 模式分析:

  • IReceiver的细节,包含行为的具体如何实施与执行。
  • Command可以理解为一个容器,用这个容器把Receiver细节包裹在里面。在执行Execute时调用具体的实现方式。
  • Invoker负责批量拆开Command包裹,来执行所含的IReceiver细节。

在PureMVC中,这种类似命令模式的实现,将一个要执行的函数包裹起来,在需要的时候拆开包裹进行调用其中的内容。

        // 关键代码片段
        public virtual void NotifyObserver(INotification notification)
        {
            object context;
            string method;

            lock (m_syncRoot)
            {
                context = NotifyContext;
                method = NotifyMethod;
            }

            Type t = context.GetType();
            BindingFlags f = BindingFlags.Instance | BindingFlags.Public | BindingFlags.IgnoreCase;
            MethodInfo mi = t.GetMethod(method, f);
            mi.Invoke(context, new object[] { notification });
        }

Observer相当于一个Command,其中NotifyObserver就是一个Command中的Execute,通过对context进行反射找到真正的类,然后根据method找到这类中的具体方法,接下来传入参数执行即可。

Controller中的ExecuteCommand方法,他直接根据类的名称反射一个全新的类,然后执行其中的Execute方法,简单明了。

        public virtual void ExecuteCommand(INotification note)
        {
            Type commandType = null;

            lock (m_syncRoot)
            {
                if (!m_commandMap.ContainsKey(note.Name)) return;
                commandType = m_commandMap[note.Name];
            }

            object commandInstance = Activator.CreateInstance(commandType);

            if (commandInstance is ICommand)
            {
                ((ICommand) commandInstance).Execute(note);
            }
        }

使用命令模式来应对这种需要排序执行的操作是首选,可以显著降低系统的耦合度,对于后期的需求变动也符合开闭原则,但命令过多容易造成命令类数量爆炸的情况。

3.3.7 应用场景:

需要排队执行或者撤销/重做等处理的需求。

3.3.8 小结:

命令模式利用对象方便排序、增减的特点进一步对函数的执行进行封装,降低系统中函数间交互的耦合度。


更多设计模式详见:设计模式全家桶

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 命令模式 现在多数应用都有撤销操作。虽然难以想象,但在很多年里,任何软件中确实都不存在撤销操作。撤销操作是在197...
    英武阅读 1,457评论 0 50
  • 目录 本文的结构如下: 什么是命令模式 为什么要用该模式 模式的结构 代码示例 优点和缺点 适用环境 模式应用 总...
    w1992wishes阅读 1,148评论 2 9
  • javascript设计模式与开发实践 设计模式 每个设计模式我们需要从三点问题入手: 定义 作用 用法与实现 单...
    穿牛仔裤的蚊子阅读 4,176评论 0 13
  • 高中时被一本《行摄四川》勾引,看着里面的照片和攻略让我决意此后要走遍里面的每一个地方。其中也包括若尔盖,大草原上一...
    竹子婧阅读 277评论 0 3
  • 但凡能想到的,都不会发生,像我这样整天想东想西的人,老天爷也是很无奈了。光是想我想不到的事情就需要好几年了吧,我原...
    舟初阅读 116评论 0 0