6.3.2 在 C# 中处理计划
在 C# 中,我们构建的 MapSchedule 方法,类似于 F# 中的 mapSchedule 函数;另外,它有两个参数:一个是用于计算新日期的函数,另一个是原始计划。因为我们将在 C# 中使用可选值,需要 switch 块和 Tag 属性,这我们在第五章已经看到过。清单6 .9显示了完整的实现。
Listing 6.9 schedule 类型的映射操作 (C#)
public static Schedule MapSchedule
(this Schedule schedule, Func<DateTime, DateTime> rescheduleFunc){ [1]
switch(schedule.Tag) { 使用 this 修饰符
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,再把它传递给计算新日期的操作。
当有几个类似的要执行的操作时,我们直接使用计划类型,可能是十分乏味,因为,我们必须为每个操作,多次提供相同的展开和包装代码。在本节,我们看到了设计良好的高阶函数,可以简化处理很多值。现在,我们看一下用高阶函数处理另一种可选值,在第五章介绍过的选项(option)类型。