Как установить десятичные разделители в контроллерах ASP.NET MVC?

Я работаю с приложением NerdDinner, пытаясь научиться ASP.NET MVC. Однако я столкнулся с проблемой глобализации, когда мой сервер представляет числа с плавающей запятой с запятой в качестве десятичного разделителя, но для карты Virtual Earth они требуются с точками, что вызывает некоторые проблемы.

Я уже решил проблему с отображением JavaScript в моих представлениях, но если я сейчас попытаюсь опубликовать отредактированную запись обеда с точками в качестве десятичных разделителей, контроллер завершится неудачно (выброс InvalidOperationException) при обновлении модели (в методе UpdateModel()). Я чувствую, что должен установить правильную культуру где-нибудь в контроллере, я пробовал это в OnActionExecuting(), но это не помогло.


person Pawel Krakowiak    schedule 27.04.2009    source источник


Ответы (4)


Я только что повторно рассмотрел проблему в реальном проекте и, наконец, нашел рабочее решение. Правильное решение - иметь настраиваемый связыватель модели для типа decimaldecimal?, если вы их используете):

using System.Globalization;
using System.Web.Mvc;

public class DecimalModelBinder : DefaultModelBinder
{
    public override object BindModel(ControllerContext controllerContext, ModelBindingContext bindingContext)
    {
        object result = null;

        // Don't do this here!
        // It might do bindingContext.ModelState.AddModelError
        // and there is no RemoveModelError!
        // 
        // result = base.BindModel(controllerContext, bindingContext);

        string modelName = bindingContext.ModelName;
        string attemptedValue = bindingContext.ValueProvider.GetValue(modelName)?.AttemptedValue;

        // in decimal? binding attemptedValue can be Null
        if (attemptedValue != null)
        {
            // Depending on CultureInfo, the NumberDecimalSeparator can be "," or "."
            // Both "." and "," should be accepted, but aren't.
            string wantedSeperator = NumberFormatInfo.CurrentInfo.NumberDecimalSeparator;
            string alternateSeperator = (wantedSeperator == "," ? "." : ",");

            if (attemptedValue.IndexOf(wantedSeperator, StringComparison.Ordinal) == -1
                && attemptedValue.IndexOf(alternateSeperator, StringComparison.Ordinal) != -1)
            {
                attemptedValue = attemptedValue.Replace(alternateSeperator, wantedSeperator);
            }

            try
            {
                if (bindingContext.ModelMetadata.IsNullableValueType && string.IsNullOrWhiteSpace(attemptedValue))
                {
                    return null;
                }

                result = decimal.Parse(attemptedValue, NumberStyles.Any);
            }
            catch (FormatException e)
            {
                bindingContext.ModelState.AddModelError(modelName, e);
            }
        }

        return result;
    }
}

Затем в Global.asax.cs в Application_Start ():

ModelBinders.Binders.Add(typeof(decimal), new DecimalModelBinder());
ModelBinders.Binders.Add(typeof(decimal?), new DecimalModelBinder());

Обратите внимание, что код не мой, я на самом деле нашел его в блоге Кристофа Нейринка здесь. Я только что отредактировал несколько строк и добавляю подшивку для определенного типа данных, а не заменяю подшивку по умолчанию.

person Pawel Krakowiak    schedule 25.02.2011
comment
Я только что обновил свой ответ, добавив дополнительный код проверки, чтобы он правильно обрабатывал свойства, допускающие значение NULL. Я обнаружил, что если он обнаружит пустую строку в качестве запрошенного значения, он вызовет исключение, даже если привязанное свойство имеет значение NULL. Теперь все должно работать нормально. - person Pawel Krakowiak; 02.09.2013
comment
Есть дополнительное исправление на стороне клиента, которое следует применить вместе с этим. Вы найдете его здесь: stackoverflow.com/a/8102159/41420 - person Pawel Krakowiak; 02.09.2013
comment
Вам не нужно этого делать. Что вам нужно сделать, так это установить культуру текущего потока до того, как связыватель модели будет выполнен, как в этом примере. - person NightOwl888; 11.10.2017

Установите это в своем web.config

  <system.web>
    <globalization uiCulture="en" culture="en-US" />

Похоже, вы используете сервер, на котором настроен язык, в котором используются запятые вместо десятичных знаков. Вы можете настроить язык и региональные параметры так, чтобы запятые использовались так, как разработано ваше приложение, например en-US.

person Nick Berardi    schedule 27.04.2009
comment
как насчет заголовков на странице aspx? Есть ли у кого-нибудь из них что-нибудь для культуры? - person Nick Berardi; 28.04.2009
comment
@Nick: Нет, там нет ничего особенного. - person Pawel Krakowiak; 11.05.2009

Можете ли вы проанализировать текст, используя инвариантную культуру - извините, у меня нет кода NerdDinner в fornt of me, но если вы передаете десятичные дроби, разделенные точками, то синтаксический анализ должен быть в порядке, если вы скажете ему использовать инвариантную культуру . Например.

 float i = float.Parse("0.1", CultureInfo.InvariantCulture);

Изменить. Кстати, я подозреваю, что это ошибка в коде NerdDinner, аналогичная вашей предыдущей проблеме.

person Steve    schedule 27.04.2009
comment
Нет, никаких свойств я не трогаю. Предполагается, что в UpdateModel () это произойдет волшебным образом, но он выдает не очень полезное исключение без подробностей. Есть две проблемы - при представлении данных из модели мне нужны точки, потому что отображение JS сломается, однако при возвращении данных в модель мне нужны запятые на этот раз, потому что в противном случае сломается контроллер. Это все немного странно, и я не уверен, как это решить, Google не помог, и SO тоже. :( - person Pawel Krakowiak; 28.04.2009
comment
Да, я не думаю, что вы можете решить эту проблему, не меняя код. У меня есть код ND, и я когда-нибудь посмотрю и попытаюсь проверить это и отправить вам исправление, если смогу, но боюсь, что это не скоро, так как придется работать :-) - person Steve; 28.04.2009
comment
Хорошо, я думаю, вам нужно отменить изменение, которое вы внесли в свой другой вопрос, я разместил там предложение о том, как исправить проблему, и, надеюсь, мое исправление не вызовет побочного эффекта, который вы видите здесь. - person Steve; 28.04.2009
comment
А затем вам нужно взглянуть на stackoverflow.com/questions/528545/, чтобы узнать, как даты работают (или иначе) в MVC ... - person Steve; 28.04.2009

У меня другой взгляд на это, может тебе понравится. Что мне не нравится в принятом ответе, так это то, что он не проверяет наличие других символов. Я знаю, что будет случай, когда символ валюты будет в поле, потому что мой пользователь не знает лучше. Так что да, я могу проверить javascript, чтобы удалить его, но что, если по какой-то причине javascript не включен? Тогда могут пройти лишние персонажи. Или, если кто-то попытается спамить вас, пропустив неизвестных персонажей ... кто знает! Поэтому я решил использовать регулярное выражение. Это немного медленнее, на крошечную долю медленнее - в моем случае 1000000 итераций регулярного выражения заняли чуть менее 3 секунд, в то время как около 1 секунды для замены строки в коме и точке. Но, поскольку я не знаю, какие персонажи могут пройти, я рад этому малейшему количеству ударов по производительности.

public class DecimalModelBinder : DefaultModelBinder
{
    public override object BindModel(ControllerContext controllerContext,
                                     ModelBindingContext bindingContext)
    {
        string modelName = bindingContext.ModelName;
        string attemptedValue =
            bindingContext.ValueProvider.GetValue(modelName).AttemptedValue;

        if (bindingContext.ModelMetadata.IsNullableValueType
                && string.IsNullOrWhiteSpace(attemptedValue))
        {
            return null;
        }

        if (string.IsNullOrWhiteSpace(attemptedValue))
        {
            return decimal.Zero;
        }

        decimal value = decimal.Zero;
        Regex digitsOnly = new Regex(@"[^\d]", RegexOptions.Compiled);
        var numbersOnly = digitsOnly.Replace(attemptedValue, "");
        if (!string.IsNullOrWhiteSpace(numbersOnly))
        {
            var numbers = Convert.ToDecimal(numbersOnly);
            value = (numbers / 100m);

            return value;
        }
        else
        {
            if (bindingContext.ModelMetadata.IsNullableValueType)
            {
                return null;
            }

        }

        return value;
    }
}

Как правило, удалите все символы, не являющиеся цифрами, для строки, которая не пуста. Преобразовать в десятичный. Разделить на 100. Вернуть результат.

Работает на меня.

person Colin Wiseman    schedule 11.11.2015
comment
Хотя в некоторых случаях это работает хорошо, если десятичная дробь не имеет значения 0,00 (т.е. есть проверка javascript или что-то еще, что ее убирает), метод деления на 100 не работает. - person Ben; 23.04.2017
comment
да. Я заметил это в нескольких случаях, а затем удостоверился, что мой код всегда добавляет десятичные разряды через JavaScript перед публикацией. Но в реальной жизни этого не всегда можно ожидать. - person Colin Wiseman; 23.04.2017