DateTime.Parse没有在MVC自定义绑定器上工作

时间:2016-04-27 09:59:56

标签: c# asp.net-mvc datetime cultureinfo defaultmodelbinder

我从.NET MVC Web应用程序中的Action的隐式绑定中获取DateTime。问题是我以格式" MM / dd / yyyy"来获取日期。而我通过格式为#d; dd / MM / yyyy"的Ajax发送查询字符串。

我知道这是.NET MVC Binder的一个已知问题,当使用GET协议而不是POST时,所以我确实实现了一个自定义绑定器来将日期解析为正确的格式。这是代码:

public class SearchVMBinder:DefaultModelBinder
{
    public override object BindModel(ControllerContext controllerContext, ModelBindingContext bindingContext)
    {
        SearchVM result = (SearchVM)base.BindModel(controllerContext, bindingContext);
        try
        {                
            result.Date = DateTime.Parse(result.Date.ToString(),CultureInfo.GetCultureInfo("es-ES"));
        }
        catch(Exception e)
        {
            NLog.LogManager.GetCurrentClassLogger().Error("Error al hacer el Bind específico de SearchVM. ", e);
        }

        return result;
    }
}

但是,使用该代码,Parse无法正常工作,它什么都不做。我正在测试日期,例如" 01/04/2014 11:37:00" (4月)我正在接受" result.Date"日期" 04/01/2014 11:37:00" (1月),之前和解析。

所以,问题是:为什么方法" DateTime.Parse"没有正确解析日期?

更新:

以下是SearchVM的代码:

[ModelBinder(typeof(SearchVMBinder))]
public class SearchVM
{
    public DateTime Date { get; set; }
    public string StudyCaseNumber { get; set; }
    public string PatientNumber { get; set; }
    public string PatientName { get; set; }
    public string PatientFamilyName { get; set; }
    public string PatientMothersMaidenName { get; set; }
    public string DoctorName { get; set; }
    public string RoomName { get; set; }
    public E_OrderedBy OrderBy { get; set; }

}

这里是控制器动作的标题:

public ActionResult ListSearch(SearchVM searchFilter)

谢谢。

4 个答案:

答案 0 :(得分:0)

我们走了:

尝试将您的DateTime发布/发送为UTC字符串: 你可以像这样转换:

DateTime startdate = TimeZoneInfo.ConvertTimeToUtc(input.startdate.Value);   //Converting to UTC time from local time

希望它有所帮助;)

答案 1 :(得分:0)

这样的事情会起作用

public override object BindModel(ControllerContext controllerContext, ModelBindingContext bindingContext)
    {
        var testFormat = bindingContext.ModelMetadata.DisplayFormatString;
        var value = bindingContext.ValueProvider.GetValue(bindingContext.ModelName);

        if (!string.IsNullOrEmpty(testFormat) && value != null)
        {
            DateTime testDate;
            testFormat = testFormat.Replace("{0:", string.Empty).Replace("}", string.Empty);
            // use the format specified in the testFormat attribute to parse the date
            if(DateTime.TryParseExact(value.AttemptedValue, testFormat, new System.Globalization.CultureInfo("es-ES"), DateTimeStyles.None, out testDate);)
            {
                return testDate;
            }
            else
            {
                //if you want allow nulls
                //date = DateTime.Now.Date;
                //return date;

                bindingContext.ModelState.AddModelError(
                    bindingContext.ModelName,
                    string.Format("{0} is an invalid date format", value.AttemptedValue)
                );
            }
        }

        return base.BindModel(controllerContext, bindingContext);
    }

答案 2 :(得分:0)

您的代码正常运作。

请参阅此问题的最佳答案:Convert DateTime from English to Spanish

你说:

  

但是,使用该代码,Parse无法正常工作,它什么都不做。   我正在测试日期,例如" 01/04/2014 11:37:00" (四月)和我   得到"结果。日期"日期" 04/01/2014 11:37:00" (一月),   之前和解析。

这是一种非常特殊的工作方式 - 这种方式是区域设置不同的结果。您使用了" es-ES"格式。

https://www.bjelic.net/2011/01/26/coding/formatting-date-time-currency-in-various-cultures/

result.Date = 

    DateTime.Parse(result.Date.ToString(),
    CultureInfo.GetCultureInfo("es-ES"));

结果日期是根据您" 01/04/2014 11:37:00"并且es-ES区域设置是dd / MM / yyyy,因此您当前正在使用英语,它会进行转换。

答案 3 :(得分:0)

好的,由于某种原因,“Parse”方法没有正确解释传入的DateTime格式,即“MM / dd / yyyy H:mm:ss”,所以我不得不使用“ParseExact”方法来指定它

现在,这适用于西班牙文化信息:

public override object BindModel(ControllerContext controllerContext, ModelBindingContext bindingContext)
    {
        SearchVM result = (SearchVM)base.BindModel(controllerContext, bindingContext);
        try
        {
            if (result != null)
            {
                if (result.NullableDate != null)
                {                                      
                    result.NullableDate = DateTime.ParseExact(result.NullableDate.ToString(), "MM'/'dd'/'yyyy H:mm:ss", new CultureInfo("es-ES"));
                }
            }

        }
        catch(Exception e)
        {
            NLog.LogManager.GetCurrentClassLogger().Error("Error al hacer el Bind específico de SearchVM. ", e);
        }

        return result;
    }