我有一个数据库模型,它将设置存储为 32 位整数上的位.HasFlag
。前端使用 AngularJs (1),遗憾的是它不允许在表达式中进行按位运算。因此,我创建了一个扩展方法,用于从枚举转换为每个标志的字典以及它是否打开:
public static IDictionary<TEnum, bool> GetAllFlags<TEnum>(this TEnum obj) where TEnum: Enum
{
return Enum.GetValues(typeof(TEnum))
.Cast<TEnum>()
.ToDictionary(flag => flag, flag => obj.HasFlag(flag));
}
我还创建了一个扩展方法来设置值,但它依赖于其他扩展方法,并且不会被调用(如下所述),因此为了简洁起见,我将省略它们。
这是正在使用的枚举的示例(请注意,它的类型默认为int并且只有很少的选项,但它确实定义了一个0奇怪的类型)
public enum Settings
{
None = 0,
Setting1 = 1,
Setting2 = 2,
Setting3 = 4,
}
所有这些都作为对象的属性公开,如下所示
// SettingsFlags comes from another partial and is a Settings enum persisted to the database as an `INT`
public parital class Options
{
IDictionary<Settings, bool> Flags {
get { return SettingsFlags.GetAllFlags(); }
set { SettingsFlags.SetAllFlags(value); } // this never gets called
}
}
JSON当为客户端序列化时,这非常有效。但是,当在请求中收到它时,它会抛出以下异常:
System.InvalidCastException: Specified cast is not valid.
at System.Web.Mvc.DefaultModelBinder.CollectionHelpers.ReplaceDictionaryImpl[TKey,TValue](IDictionary`2 dictionary, IEnumerable`1 newContents)
当进入调试器时,该get块被调用并返回预期的字典。之后,调试器中无需执行任何其他步骤即可引发错误。
以下是端点的示例(不符合 REST 标准):
public JsonResult UpdateSingleItem(Options options, ...)
{ // breakpoint here is never called.
....
}`
使用ReSharperwithDotPeak检查代码为我提供了引发错误的方法的以下信息:
// System.Web.Mvc.DefaultModelBuilder.CollectionHelpers
private static void ReplaceDictionaryImpl<TKey, TValue>(
IDictionary<TKey, TValue> dictionary,
IEnumerable<KeyValuePair<object, object>> newContents)
{
dictionary.Clear();
foreach (KeyValuePair<object, object> newContent in newContents)
{
TKey key = (TKey) newContent.Key;
TValue obj = newContent.Value is TValue ? (TValue) newContent.Value : default (TValue);
dictionary[key] = obj;
}
}
小怪兽爱吃肉
相关分类