当我的模型有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"
参数,并将 直接分配给操作中的属性。啊。子>
答案 0 :(得分:3)
非常简单,真的。如果DefaultModelBinder
实例非空,IEnumerable<>
将不会覆盖它。如果它为null,它将创建一个新的List<T>
并填充它。
如果它是非null,则它具有某些类型的列表,它知道如何处理。如果您的列表实现ICollection<>
,那么它将填充它。但是你的实例(yield
)根本无法更新!
如果您对覆盖foobacking
感到满意,那么您可以通过编写自定义模型绑定器来解决此问题。