注意:这个问题已经超过9年了!

您最好的选择是搜索更新的问题,或者搜索下面的答案,寻找您特定的MVC版本,因为这里的许多答案现在已经过时了。

如果你确实找到了适合你的版本的答案,请确保答案包含了你正在使用的MVC版本。 (原来的问题从下面开始)


这对我来说似乎有点奇怪,但据我所知,这就是你怎么做的。

我有一个对象集合,我希望用户从中选择一个或多个对象。这对我来说是“带复选框的表单”。我的对象没有任何“选定”的概念(它们是通过反序列化wcf调用形成的基本POCO)。所以,我这样做:

public class SampleObject{
  public Guid Id {get;set;}
  public string Name {get;set;}
}

在视图中:

<%
    using (Html.BeginForm())
    {
%>
  <%foreach (var o in ViewData.Model) {%>
    <%=Html.CheckBox(o.Id)%>&nbsp;<%= o.Name %>
  <%}%>
  <input type="submit" value="Submit" />
<%}%>

并且,在控制器中,这是我能看到的唯一方法来找出用户检查了什么对象:

public ActionResult ThisLooksWeird(FormCollection result)
{
  var winnars = from x in result.AllKeys
          where result[x] != "false"
          select x;
  // yadda
}

首先,它很怪异,其次,对于用户选中的那些项,FormCollection将其值列出为“true false”而不仅仅是true。

显然,我遗漏了一些东西。我认为这是基于这样的想法构建的,即在html表单中所作用的集合中的对象是使用UpdateModel()或通过ModelBinder更新的。

但我的对象并不是为此而设置的;这意味着这是唯一的办法吗?还有别的办法吗?


当前回答

当使用HtmlHelper的复选框时,我更喜欢使用已发布的复选框表单数据作为数组。我真的不知道为什么,我知道其他方法也有用,但我想我只是更喜欢把逗号分隔的字符串尽可能地当作一个数组。

所以做一个“检查过的”或真实的测试应该是:

//looking for [true],[false]
bool isChecked = form.GetValues(key).Contains("true"); 

做虚假检查会是:

//looking for [false],[false] or [false]
bool isNotChecked = !form.GetValues(key).Contains("true"); 

主要的区别是在返回数组时使用GetValues。

其他回答

超文本标记语言CheckBox做了一些奇怪的事情——如果你在结果页面上查看源代码,你会看到每个复选框旁边都生成了一个<input type="hidden" />,这解释了你看到的每个表单元素的"true false"值。

试试这个,它肯定能在ASP上工作。NET MVC Beta版,因为我刚刚尝试过。

把它放在视图中,而不是使用Html.CheckBox():

<% using (Html.BeginForm("ShowData", "Home")) {  %>
  <% foreach (var o in ViewData.Model) { %>
    <input type="checkbox" name="selectedObjects" value="<%=o.Id%>">
    <%= o.Name %>
  <%}%>
  <input type="submit" value="Submit" />
<%}%>

您的复选框都被称为selectedObjects,每个复选框的值是对应对象的GUID。

然后发布到下面的控制器动作(或做一些有用的事情而不是Response.Write())

public ActionResult ShowData(Guid[] selectedObjects) {
    foreach (Guid guid in selectedObjects) {
        Response.Write(guid.ToString());
    }
    Response.End();
    return (new EmptyResult());
}

这个例子只会写你选中的方块的guid;ASP。NET MVC将选中复选框的GUID值映射到GUID [] selectedObjects参数中,甚至解析来自请求的字符串。将集合形成实例化的GUID对象,我认为这是相当不错的。

像这样的怎么样?

bool isChecked = false;
if (Boolean.TryParse(Request.Form.GetValues(”chkHuman”)[0], out isChecked) == false)
    ModelState.AddModelError(”chkHuman”, “Nice try.”);

与nautic20的答案相同,只是简单地使用MVC默认模型绑定复选框列表与ViewModel中string/int/enum的集合属性同名。就是这样。

但有一个问题需要指出。在每个复选框组件中,不应该在其中放入“Id”,这会影响MVC模型绑定。

下面的代码将用于模型绑定:

 <% foreach (var item in Model.SampleObjectList)
       { %>
        <tr>
            <td><input type="checkbox" name="SelectedObjectIds" value="<%= item.Id%>" /></td>
            <td><%= Html.Encode(item.Name)%></td>
        </tr>
 <% } %>

下面的代码将不会绑定到模型(这里的区别是为每个复选框分配id)

<% foreach (var item in Model.SampleObjectList)
       { %>
        <tr>
            <td><input type="checkbox" name="SelectedObjectIds" id="[some unique key]" value="<%= item.Id%>" /></td>
            <td><%= Html.Encode(item.Name)%></td>
        </tr>
<% } %>

HtmlHelper添加了一个隐藏输入来通知控制器未检查状态。 所以要有正确的检查状态:

bool bChecked = form[key].Contains("true");

这个问题也出现在1.0版中。Html.Checkbox()导致添加另一个隐藏字段,其名称/id与原始复选框相同。当我尝试使用document.GetElemtentsByName()加载复选框数组时,您可以猜到事情是如何变得混乱的。这很奇怪。