我试图找到一个更好的方法来处理一些不断增长的if构造来处理不同类型的类.这些类最终是围绕不同的值类型(int,DateTime等)包装一些额外的状态信息.所以这些类之间的主要区别是它们包含的数据类型.虽然它们实现通用接口,但它们也需要保存在同构的集合中,因此它们还实现了非通用接口.类实例根据它们所代表的数据的类型来处理,并且它们的传播继续或不会继续.
虽然这不一定是.NET或C#问题,我的代码在C#中.
示例类:
interface ITimedValue { TimeSpan TimeStamp { get; } } interface ITimedValue<T> : ITimedValue { T Value { get; } } class NumericValue : ITimedValue<float> { public TimeSpan TimeStamp { get; private set; } public float Value { get; private set; } } class DateTimeValue : ITimedValue<DateTime> { public TimeSpan TimeStamp { get; private set; } public DateTime Value { get; private set; } } class NumericEvaluator { public void Evaluate(IEnumerable<ITimedValue> values) ... }
我想出了两个选择:
双发
我最近才了解到访客模式及其使用双重调度来处理这种情况.这种呼吁是因为它会允许不需要的数据不能传播(如果我们只想处理一个int,我们可以处理与DateTime的不同).此外,处理不同类型的行为将仅限于处理调度的单个类.但是,如果/当需要支持新的值类型时,会有一些维护.
联盟班
包含每个支持的值类型的属性的类可以是这些类存储的每个类型.对值的任何操作将影响适当的组件.这不仅仅是双重调度策略的复杂性和维护性,而是意味着每一块数据都会不必要地传播,因为你不能再根据这种数据类型进行区分”.然而,如果/当需要支持新类型时,它们只需要进入这个类(加上需要创建以支持新数据类型的其他类).
class UnionData { public int NumericValue; public DateTime DateTimeValue; }
有更好的选择吗?在这两个选项中有一些我不认为我应该的东西吗?
解决方法
方法1,使用动态双重发送(信用到
@L_301_0@).
基本上你可以让你的访客模式简化如下:
基本上你可以让你的访客模式简化如下:
class Evaluator { public void Evaluate(IEnumerable<ITimedValue> values) { foreach(var v in values) { Eval((dynamic)(v)); } } private void Eval(DateTimeValue d) { Console.WriteLine(d.Value.ToString() + " is a datetime"); } private void Eval(NumericValue f) { Console.WriteLine(f.Value.ToString() + " is a float"); } }
使用样本:
var l = new List<ITimedValue>(){ new NumericValue(){Value= 5.1F},new DateTimeValue() {Value= DateTime.Now}}; new Evaluator() .Evaluate(l); // output: // 5,1 is a float // 29/02/2012 19:15:16 is a datetime