如何为DateTime创建和使用自定义IFormatProvider?

时间:2010-03-04 19:34:41

标签: c# iformatprovider

我试图创建一个IFormatProvider实现,它可以识别DateTime对象的自定义格式字符串。这是我的实施:

 public class MyDateFormatProvider : IFormatProvider, ICustomFormatter
 {
  public object GetFormat(Type formatType)
  {
   if (formatType == typeof(ICustomFormatter))
   {
    return this;
   }
   return null;
  }

  public string Format(string format, object arg, IFormatProvider formatProvider)
  {
   if(arg == null) throw new ArgumentNullException("arg");
   if (arg.GetType() != typeof(DateTime)) return arg.ToString();
   DateTime date = (DateTime)arg;
   switch(format)
   {
    case "mycustomformat":
     switch(CultureInfo.CurrentCulture.Name)
     {
      case "en-GB":
       return date.ToString("ddd dd MMM");
      default:
       return date.ToString("ddd MMM dd");
     }
    default:
     throw new FormatException();
   }
  } 

我希望能够在DateTime.ToString(string format, IFormatProvider provider)方法中使用它,但是:

DateTime d = new DateTime(2000, 1, 2);
string s = d.ToString("mycustomformat", new MyDateFormatProvider());

在该示例中,在美国文化中运行,结果为"00cu0Ao00or0aA",显然是因为正在解释标准的DateTime格式字符串。

但是,当我以下列方式使用同一个类时:

DateTime d = new DateTime(2000, 1, 2);
string s = String.Format(new MyDateFormatProvider(), "{0:mycustomformat}", d);

我得到了我的期望,即"Sun Jan 02"

我不明白不同的结果。有人可以解释一下吗?

谢谢!

3 个答案:

答案 0 :(得分:18)

简短的解释是,而

DateTime.ToString(string format, IFormatProvider provider)

允许您将实现IFormatProvider的任何内容作为其中一个参数传递,它实际上只支持在其代码中实现IFormatProvider的两种可能类型:

DateTimeFormatInfoCultureInfo

如果无法将参数(使用as)作为其中之一进行投放,则该方法将默认为CurrentCulture

String.Format不受此限制的限制。

答案 1 :(得分:10)

使用Reflector检查DateTime.ToString方法表明DateTime结构使用DateTimeFormatInfo.GetInstance方法来获取用于格式化的提供程序。 DateTimeFormatInfo.GetInstance从传入的提供程序请求格式化程序DateTimeFormatInfo,而不是ICustomFormmater,因此它只返回DateTimeFormatInfoCultureInfo的实例。找不到提供者。似乎DateTime.ToString方法不像ICustomFormatter方法那样遵守StringBuilder.Format接口,正如您的String.Format示例所示。

我同意DateTime.ToString方法应支持ICustomFormatter接口,但目前似乎不支持。这可能已经改变或将在.NET 4.0中发生变化。

答案 2 :(得分:3)

使用扩展方法:)

public static class FormatProviderExtension
    {
        public static string FormatIt(string format, object arg, IFormatProvider formatProvider)
        {
            if (arg == null) throw new ArgumentNullException("arg");
            if (arg.GetType() != typeof(DateTime)) return arg.ToString();
            DateTime date = (DateTime)arg;
            switch (format)
            {
                case "mycustomformat":
                    switch (CultureInfo.CurrentCulture.Name)
                    {
                        case "en-GB":
                            return date.ToString("ddd dd MMM");
                        default:
                            return date.ToString("ddd MMM dd");
                    }
                default:
                    throw new FormatException();
            }
        }

        public static string ToString(this DateTime d, IFormatProvider formatProvider, string format)
        {
            return FormatIt(format, d, formatProvider);
        }
    }