Html.CheckBoxFor类型转换错误

Html.CheckBoxFor TypeConversition Error

我有一个视图模型:

public class RegisterModel
{
   ...
   public bool Confirmation{ get; set; }
}

我在我的视图中使用复选框助手:

@model RegisterModel

......
@Html.CheckBoxFor(m => m.Confirmation) 

此复选框 html 助手创建:

<input id="Confirmation" name="Confirmation" value="true" type="checkbox">
<input name="Confirmation" value="false" type="hidden">

在控制器上

[HttpPost]
[ValidateAntiForgeryToken]
    public ActionResult Register(RegisterModel model)
{
   if (!ModelState.IsValid)
                return View(model);
.....
}

假设一些用户将输入值更改为 'xxx' 并发布。因此,模型无效,我们 return 查看。之后,Html.CheckBoxFor 给出了这个错误:

The parameter conversion from type 'System.String' to type 'System.Boolean' failed.

内部异常:

System.FormatException: xxx is not a valid value for Boolean

当我们return查看时:Model.Confirmation值为falseRequest["Confirmation"]值为'xxx' .

此错误来自 ValueProviderResult class ConvertSimpleType 方法。我认为,它试图将 Request["Confirmation"] 值转换为布尔值并给出错误。

[SuppressMessage("Microsoft.Design", "CA1031:DoNotCatchGeneralExceptionTypes", Justification = "Conversion failure is not fatal")]
private static object ConvertSimpleType(CultureInfo culture, object value, Type destinationType)
{

.....
    TypeConverter converter = TypeDescriptor.GetConverter(destinationType);
    bool canConvertFrom = converter.CanConvertFrom(value.GetType());
    if (!canConvertFrom)
    {
        converter = TypeDescriptor.GetConverter(value.GetType());
    }
    if (!(canConvertFrom || converter.CanConvertTo(destinationType)))
    {
        // EnumConverter cannot convert integer, so we verify manually
        if (destinationType.IsEnum && value is int)
        {
            return Enum.ToObject(destinationType, (int)value);
        }
             string message = String.Format(CultureInfo.CurrentCulture, MvcResources.ValueProviderResult_NoConverterExists,
                                           value.GetType().FullName, destinationType.FullName);
        throw new InvalidOperationException(message);
    }

.....
}

如何修复或避免此错误?

根据@StephenMuecke 的说法,这是默认行为。你可以查看详细的 answer

根据@ataravati 的说法,我们应该在 model.IsValid==false 上处理这个问题。如果模型无效,我们删除复选框的值并分配新的值。因此,当我们 return 查看时,我们不会收到任何错误。

 if (!ModelState.IsValid)
            {
                bool confirmation;

                bool.TryParse(Request["Confirmation"],out confirmation);
                ModelState.Remove("Confirmation");
                request.Confirmation = confirmation;
                return View(request);
            }

根据@StephenMuecke 的说法,如果复选框输入值不是布尔值,则用户肯定是恶意的。因此,我们将用户重定向到另一个具有 tracking/blocking ip 算法和 returns 404 作为视图的操作。

  if (!ModelState.IsValid)
            {
                bool confirmation;
                if (bool.TryParse(Request["Confirmation"], out confirmation))
                    return View(request);
                return RedirectToAction("Http404", "Errors"); //This not just redirecting 404, it has also tracking/blocking ip algorithm.
            }