zoukankan      html  css  js  c++  java
  • Swift语言之命令模式(Command Pattern)实现

    今天遇到这样一个问题,我现在有一个整数数组,如:

    var numbers = [3, 7, 12, 9, 200]

    现需要对其中的每一个数字都执行一系列相同的加减乘除操作,如对每一个数字都加5乘8再减去1,但是这样的操作在编译时并不确定,需要在运行时由用户指定;

    一看到这个题目,当然就想到了用设计模式中的命令模式来实现;

    于是先写了这样的一个类:

    class Calculator {
    private(set)
    var total = 0 required init(value: Int){ total = value } func add(amount: Int) { total += amount } func substract(amount: Int) { total -= amount } func multiply(amount: Int) { total = total * amount } func divide(amount: Int) { total = total / amount } }

    这个类用于实现对某个数执行不同的操作。

    下一步中我创建了一个Command类,用于记录需要执行的操作:

    struct Command {
        
        typealias OperationType = (Calculator -> Int -> Void)
        
        let operation: OperationType
        let amount: Int
        
        init(operation: OperationType, amount: Int) {
            self.operation = operation
            self.amount = amount
        }
        
        func execute(calculator: Calculator) {
            operation(calculator)(amount)
        }
    }

    在该类中我定义了一个名为OperationType的类型别名,从定义中可以看出,OperationType类型的实例是一个Closure,该Closure接受一个Calculator类型的实例,并返回一个类型为Int -> Void的Closure(此Closure的类型就是Calculator中每个方法的类型);

    amount为执行那个操作的参数值;

    待一切完毕后,由用户来指定要执行的命令,如将数组numbers中的每一个值加5乘8再减去1,则可以这样实现:

    var commands = [Command]()
    commands.append(Command(operation: Calculator.add, amount: 5))
    commands.append(Command(operation: Calculator.multiply, amount: 8))
    commands.append(Command(operation: Calculator.substract, amount: 1))

    这样做并没有达到Command设计模式想要达到的目的,及实现命令的发出者与命令的执行者之间的脱耦;于是我加入了一个新的enum类型:

    enum OperationType {
        case Add(Int)
        case Substract(Int)
        case Multiply(Int)
        case Divide(Int)
    }

    这个enum类型中的每一个成员都自带一个关联值,及要执行操作的参数,于是Command类型可以改变为:

    struct Command {
        
        let operationType: OperationType
        
        init(operationType: OperationType) {
            self.operationType = operationType
        }
        
        func execute(calculator: Calculator) {
            switch self.operationType {
                case .Add(let value):
                    calculator.add(value)
                case .Substract(let value):
                    calculator.substract(value)
                case .Multiply(let value):
                    calculator.multiply(value)
                case .Divide(let value):
                    calculator.divide(value)
            }
        }
    }

    重新创建Commands数组:

    var commands = [Command]()
    commands.append(Command(operationType: .Add(5)))
    commands.append(Command(operationType: .Multiply(8)))
    commands.append(Command(operationType: .Substract(1)))

    OK,执行所有命令:

    for number in numbers {
        var calculator = Calculator(value: number)
        
        for command in commands {
            command.execute(calculator)
        }
        
        println("(number) -> (calculator.total)")
    }

    这里是结果:

  • 相关阅读:
    bzoj 3930: [CQOI2015]选数
    bzoj 2301: [HAOI2011]Problem b
    HDU 1695 GCD
    2017ACM/ICPC广西邀请赛-重现赛 1007.Duizi and Shunzi
    2017ACM/ICPC广西邀请赛-重现赛 1010.Query on A Tree
    2017ACM/ICPC广西邀请赛-重现赛 1004.Covering
    P3501 [POI2010]ANT-Antisymmetry
    P1171 售货员的难题
    P3385 【模板】负环
    P1659 [国家集训队]拉拉队排练
  • 原文地址:https://www.cnblogs.com/cdutedu/p/4339685.html
Copyright © 2011-2022 走看看