我正在使用NerdDinner应用程序试图自学ASP.NET MVC。但是,我偶然发现了全球化问题,我的服务器使用逗号作为小数分隔符显示浮点数,但虚拟地球地图需要点数,这会导致一些问题。
我已经解决了the issue with the mapping JavaScript in my views,但是如果我现在尝试发布带有点作为十进制分隔符的已编辑的晚餐条目,则更新模型时控制器会失败(抛出InvalidOperationException
)(在{{1 metod)。我觉得我必须在控制器的某个地方设置正确的文化,我在UpdateModel()
中尝试过,但这没有帮助。
答案 0 :(得分:58)
我刚刚在一个真实的项目中重新审视了这个问题,最后找到了一个有效的解决方案。正确的解决方案是为decimal
类型设置自定义模型绑定器(如果您正在使用它们,则为decimal?
):
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;
}
}
然后在Application_Start()的Global.asax.cs中:
ModelBinders.Binders.Add(typeof(decimal), new DecimalModelBinder());
ModelBinders.Binders.Add(typeof(decimal?), new DecimalModelBinder());
请注意,代码不是我的,我实际上是在Kristof Neirynck的博客here找到的。我刚编辑了几行,并为特定数据类型添加了binder,而不是替换默认的binder。
答案 1 :(得分:5)
在web.config
中设置此项 <system.web>
<globalization uiCulture="en" culture="en-US" />
您似乎正在使用使用逗号而不是小数位的语言设置的服务器。您可以将文化调整为使用逗号设置的文化,例如en-US。
答案 2 :(得分:0)
你可以使用不变文化来解析文本 - 抱歉,我没有NerdDinner代码,但是如果你传递的是点分隔小数,那么如果你告诉它使用解析应该没问题不变的文化。 E.g。
float i = float.Parse("0.1", CultureInfo.InvariantCulture);
修改即可。我怀疑这是NerdDinner代码中的一个错误,就像你之前的问题一样。
答案 3 :(得分:0)
我对此有不同看法,你可能会喜欢它。我不喜欢接受的答案是它没有检查其他字符。我知道有一种情况,货币符号将在框中,因为我的用户不知道更好。所以,是的,我可以检查javascript删除它,但如果由于某种原因javascript没有打开怎么办?然后额外的角色可能会通过。或者,如果有人试图通过垃圾邮件传递未知字符...谁知道!所以我决定使用正则表达式。它有点慢,微小的一小部分 - 对于我的情况来说,1000,000次正则表达式的迭代只用了不到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.返回结果。
适合我。