在Command Pattern中获取执行方法的结果

时间:2012-01-14 14:39:35

标签: c# design-patterns architecture command

目前我正在尝试在一个简单的测试项目中实现事务脚本模式(正如Martin Fowler described如何使用Command Pattern),一切正常,问题是我不在知道如何在从ICommand接口继承的具体类中执行指定的方法时获取结果。

让我们向您展示一些代码,以阐明我的功能。 我有一个简单的CalculateSalaryCommand类,它继承自ICommand接口

public class CalculateSalaryCommand : ICommand
{
    private readonly CalculateSalaryTS _salaryTs;
    private readonly int _hour;
    private readonly int _salaryPerHour;

    public CalculateSalaryCommand(CalculateSalaryTS salaryTs, int hour, int salaryPerHour)
    {
        _salaryTs = salaryTs;
        _hour = hour;
        _salaryPerHour = salaryPerHour;
    }

    public void Execute()
    {
        _salaryTs.CalculateSalary(_hour, _salaryPerHour);
    }
}

和一个名为CalculateSalaryTS的简单事务脚本类

public class CalculateSalaryTS {
    public void CalculateSalary(int _hour, int _salaryPerHour) {
        Result = _hour * _salaryPerHour;
    }
}

如您所见,我将实例传递给具体的命令类,然后在Execute方法内部执行该实例的操作。好吧,一切都很好看。但有一个问题我无法返回执行方法的结果,应该是一个整数。 为了解决这个问题,我决定在Transaction Script层添加一些代码,每个事务应该从一个通用ITransactionResult接口继承,如下所示:

public interface ITransactionResult<TResult>
{
    TResult Result { get; set; }
}

然后CalculateSalaryTS类变成这样:

public class CalculateSalaryTS : ITransactionResult<Int32> {

    public void CalculateSalary(int _hour, int _salaryPerHour) {
        Result = _hour * _salaryPerHour;
    }

    public int Result { get; set; }

}

用法:

    var script = new CalculateSalaryTS();
    var command = new CalculateSalaryCommand(script, 10, 20);           
    command.Execute();
    Console.WriteLine("Salary is {0}", script.Result);

我知道这种方式有其自身的局限性但我没有任何选择,除非你给我另一个想法来处理这种情况。

提前致谢。

2 个答案:

答案 0 :(得分:22)

如果您在命令执行后绝对需要立即获得结果,则可以将结果存储在命令对象中:

public interface ICommandWithResult<T> : ICommand
{
  T Result { get; }
}

public class CalculateSalaryCommand : ICommandWithResult<int>
{
  public int Result { get; private set; }

  // ...

  public void Execute()
  {
    _salaryTs.CalculateSalary(_hour, _salaryPerHour);
    this.Result = _salaryTs.Result;
  }
}

// Usage:

var command = new CalculateSalaryCommand(new CalculateSalaryTS(), 10, 20);
command.Execute();
Console.WriteLine("Salary is {0}", command.Result);

答案 1 :(得分:5)

我认为你正在寻找能够返回结果的命令模式。

所以,

    interface ICommand<TResult>
    {
       TResult Execute();
    }


public class CalculateSalaryCommand : ICommand<int>
{
    private readonly CalculateSalaryTS _salaryTs;
    private readonly int _hour;
    private readonly int _salaryPerHour;

    public CalculateSalaryCommand(CalculateSalaryTS salaryTs, int hour, int salaryPerHour)
    {
        _salaryTs = salaryTs;
        _hour = hour;
        _salaryPerHour = salaryPerHour;
    }

    public int Execute()
    {
        return _salaryTs.CalculateSalary(_hour, _salaryPerHour);
    }
}