MVC模型绑定:为什么我不能绑定到迭代器属性?

时间:2011-02-08 20:51:24

标签: asp.net-mvc iterator ienumerable model-binding

当我的模型有IEnumerable<T>属性实现为iterator(即yield return)时,当传入的值使用方括号时,MVC的DefaultModelBinder无法绑定到该属性语法(例如"Foo[0]")。

示例模型:

namespace ModelBinderTest
{
    using System.Collections.Generic;
    public class MyModel
    {
        private List<string> fooBacking = new List<string>();
        public IEnumerable<string> Foo
        {
            get
            {
                foreach (var o in fooBacking)
                {
                    yield return o; // <-- ITERATOR BREAKS MODEL BINDING
                }
            }
            set { fooBacking = new List<string>(value); }
        }

        private List<string> barBacking = new List<string>();
        public IEnumerable<string> Bar
        {
            get
            {
                // Returning any non-iterator IEnumerable works here. Eg:
                return new List<string>(barBacking);
            }
            set { barBacking = new List<string>(value); }
        }
    }
}

失败的例子 1

namespace ModelBinderTest
{
    using System;
    using System.Linq;
    using System.Web.Mvc;
    using Microsoft.VisualStudio.TestTools.UnitTesting;

    [TestClass]
    [CLSCompliant(false)]
    public class DefaultModelBinderTestIterator
    {
        [TestMethod]
        public void BindsIterator()
        {
            // Arrange
            var model = new MyModel();

            ModelBindingContext bindingContext = new ModelBindingContext()
            {
                FallbackToEmptyPrefix = true,
                ModelMetadata = ModelMetadataProviders
                                  .Current
                                  .GetMetadataForType(null, model.GetType()),
                ModelName = "",
                ValueProvider = new NameValueCollectionValueProvider(
                    new System.Collections.Specialized.NameValueCollection()
                        {
                            { "Foo[0]", "foo" },
                            { "Bar[0]", "bar" },
                        },
                    System.Globalization.CultureInfo.InvariantCulture
                )
            };

            DefaultModelBinder binder = new DefaultModelBinder();

            // Act
            MyModel updatedModel = (MyModel)binder.BindModel(
                                     new ControllerContext(), bindingContext);

            // Assert
            Assert.AreEqual(1, updatedModel.Bar.Count(),
                            "Bar property should have been updated");
            Assert.AreEqual("bar", updatedModel.Bar.ElementAtOrDefault(0),
                            "Bar's first element should have been set");

            Assert.AreEqual(1, updatedModel.Foo.Count(),
                            "Foo property should have been updated");
            Assert.AreEqual("foo", updatedModel.Foo.ElementAtOrDefault(0),
                            "Foo's first element should have been set");
        }

    }
}

上面的单元测试会将模型的Bar属性更新为["bar"]没有问题(在集合键中有或没有方括号),但是无法将任何内容绑定到{{ 1}}属性。

有没有人知道(在较低级别)为什么实现Foo属性作为迭代器会导致模型绑定失败?

我对解决方法 2 并不感兴趣,而是对一些分析感兴趣,因为我已经用尽了我对框架的了解,远远不够;)


1:单元测试是分离SO问题的最简单方法,而不是通过整个MVC应用程序示例。

2:例如,我知道如果我从输入中删除方括号并为所有值重用相同的IEnumerable键,则模型绑定将起作用。然而,真正的失败案例需要方括号,因为集合中的每个项目都是具有自己的子属性的复杂类型。或者另一种解决方法:向操作添加非迭代器"Foo"参数,并将 直接分配给操作中的属性。啊。

1 个答案:

答案 0 :(得分:3)

非常简单,真的。如果DefaultModelBinder实例非空,IEnumerable<>将不会覆盖它。如果它为null,它将创建一个新的List<T>并填充它。

如果它是非null,则它具有某些类型的列表,它知道如何处理。如果您的列表实现ICollection<>,那么它将填充它。但是你的实例(yield)根本无法更新!

如果您对覆盖foobacking感到满意,那么您可以通过编写自定义模型绑定器来解决此问题。