6.3.2 在 C# 中处理计划

来源:互联网 发布:淘宝网开店教程视频 编辑:程序博客网 时间:2024/06/16 17:46

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)类型。

0 0