目前我正在尝试在一个简单的测试项目中实现事务脚本模式(正如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);
我知道这种方式有其自身的局限性但我没有任何选择,除非你给我另一个想法来处理这种情况。
提前致谢。
答案 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);
}
}