6.3.2 在 C# 中处理计划
在 C# 中,我们将构建 MapSchedule 方法,类似于 F# 中的 mapSchedule 函数。再次,这将有两个参数:一个用于计算新日期的函数,和原先计划。因为我们将在 C# 中处理可选值,使用 switch 块和 Tag 属性,正如在第 5 章看到的一样。清单6 .9显示了完整的实现。
Listing 6.9 Map operation for schedule type (C#)
public static Schedule MapSchedule
(this Schedule schedule, Func rescheduleFunc) {
switch(schedule.Tag) {
case ScheduleType.Never:
return new Never();
case ScheduleType.Once:
var os = (Once)schedule;
return new Once(rescheduleFunc(os.EventDate));
case ScheduleType.Repeatedly:
var rs = (Repeatedly)schedule;
DateTime newStart = rescheduleFunc(rs.StartDate);
return new Repeatedly(newStart, rs.Interval);
default:
throw new InvalidOperationException();
}
}
该方法提供了对每个可能的表示了一个分支,并在每个分支中返回一个新值。当选项携带一个可以处理的日期(Once 和 Repeatedly),它首先把这个参数值转换为相应的类型,然后,使用 rescheduleFunc 参数值去计算新的日期。
该方法实现为在 ScheduleUtils 类的内部扩展方法(为了简单起见,清单中未包括这个类的声明)。这意味着,我们可以把它,作为一个静态方法,但是,在 Schedule 类的任何实例上使用点表示法,也更具可读性。以下代码段显示了如何把列表中每个计划推迟一周:
schedules.Select(schedule =>
schedule.MapSchedule(dt => dt.AddDays(7.0)) )
这是类似于我们前面的 F# 代码。我们使用 LINQ 的 Select 方法(代替 List.map 函数),为原来列表中的每个计划,去计算新的计划。在 lambda 函数内 ,在每个原来的计划上,调用 MapSchedule,向它传递计算新日期的操作。
当我们有几个类似的要执行的操作时,直接使用这个计划类型,可能是十分乏味,因为,我们必须为每个操作,多次提供相同的展开和包装代码。在本节中,我们看到一个精心设计的高阶函数,可以简化处理很多值。现在,让我们看一下写高阶函数,来处理另一个可选值,我们在第 5 章中介绍过的 选项(option)类型。