5

我需要能够调用一个方法并传入一个未知类型的对象,然后调用正确的重载。object我还需要一个接受作为其参数类型的默认实现 。我所看到的是默认重载是唯一被使用过的重载。

这是我正在尝试做的事情的要点:

class Formatter
{
  private object Value;

  public Formatter(object val){
    Value = val;
  }

  public override string ToString()
  {
    return Format(Value);
  }

  private string Format(object value)
  {
    return value.ToString();
  }

  private string Format(DateTime value)
  {
    return value.ToString("yyyyMMdd");
  }
}

好的,到目前为止一切顺利。现在我希望能够做到这一点:

public static class FancyStringBuilder()
{
  public static string BuildTheString()
  {
    var stringFormatter = new Formatter("hello world");
    var dateFormatter = new Formatter(DateTime.Now);
    return String.Format("{0} {1}", stringFormatter, dateFormatter);
  }
}

结果FancyStringBuilder.BuildTheString()"hello world 2012-12-21 00:00:00.000",当我预期"hello world 20121221"

问题是DateTime没有调用接受 a 的重载,而是默认为接受 a 的重载object如何在不使用凌乱的 switch 语句的情况下调用正确的方法?

4

2 回答 2

1

我认为这是因为类构造函数接受一个objectas 参数,然后将该对象分配给Value也是一个object. 在那里调用 Format(object) 因为Value是类型object

试试这个

public override string ToString()
{
    if(Value is DateTime)
      return Format(Convert.ToDateTime(Value)); //this should call the right method

    return Format(Value); //works for other non-custom-format types e.g. String
}
于 2012-12-19T22:57:17.827 回答
1

在中,总是调用Formatter.ToString()覆盖。这是因为重载决议发生在编译时,而不是运行时。在编译时,唯一知道的是它是一个对象。Formatter.Format(object)Value

如果你真的想区分传入的类型,你需要在Formatter's 的构造函数中这样做。ToString()在这种情况下,您可以立即调用并仅存储格式化的结果,而不是挂在对象上:

class Formatter
{
    string formattedValue;

    public Formatter(object value)
    {
        formattedValue = value.ToString();
    }

    public Formatter(DateTime value)
    {
        formattedValue = value.ToString("yyyyMMdd");
    }

    public string ToString()
    {
        return formattedValue;
    }
}

请注意,这确实假设您的对象在创建Formatter对象和Formatter.ToString()调用时间之间没有变化,或者至少可以在Formatter创建时对字符串表示进行快照。

这也假设您在编译时知道传入的类型。如果您想要一个真正仅运行时的解决方案,则必须使用“is”运算符或typeof()比较。

如果您的目标只是提供基于传入类型的自定义 ToString() 格式,我可能会使用从类型映射到格式字符串的列表来实现:

static class Formatter
{
    private static List<Tuple<Type, string>> Formats;

    static Formatter()
    {
        Formats = new List<Tuple<Type, string>>();

        // Add formats from most-specific to least-specific type.
        // The format string from the first type found that matches
        // the incoming object (see Format()) will be used.
        AddMapping(typeof(DateTime), "yyyyMMdd");
        // AddMapping(typeof(...), "...");
    }

    private static void AddMapping(Type type, string format)
    {
        Formats.Add(new Tuple<Type, string>(type, format));
    }

    public static string Format(object value)
    {
        foreach (var t in Formats)
        {
            // If we find a type that 'value' can be assigned to
            // (either the same type, a base type, or an interface),
            // consider it a match, and use the format string.
            if (t.Item1.IsAssignableFrom(value.GetType()))
            {
                return string.Format(t.Item2, value);
            }
        }

        // If we didn't find anything, use the default ToString()...
        return value.ToString();
    }
}

这样,调用代码如下所示:

Console.WriteLine(
    "{0} {1}",
    Formatter.Format(DateTime.Now),
    Formatter.Format("banana"));
于 2012-12-20T00:12:43.257 回答