我有一个IDictionary字段,我想通过类型为IDictionary< string,dynamic>的属性公开转换是非常困难的,因为我不知道我可以.Cast<>()的IDictionary到.
我有最好的
- IDictionary properties;
- protected virtual IDictionary<string,dynamic> Properties {
- get {
- return _properties.Keys.Cast<string>()
- .ToDictionary(name=>name,name=> _properties[name] as dynamic);
- }
- }
解决方法
如果IDictionary的底层类型不实现IDictionary< string,dynamic>,则不能转换对象(period).如果是这样,一个简单的转换(IDictionary< string,dynamic>)localVar就足够了.
如果不是,你可以做两件事情:
>将IDictionary复制到通用类型.
>构建一个接受IDictionary作为依赖项的Adapter类,并实现所需的通用IDictionary,将调用从一个映射到另一个.
编辑:您刚刚发布的示例代码将在每次调用时复制字典!稍后我会用一些建议的代码再次编辑.
选项1
您的示例代码方法是复制数据的一种手段,但副本应该被缓存,或者你将要复制很多次.我建议您将实际的翻译代码放在一个单独的方法中,并在首次使用时将其从属性中调用.例如:
- private IDictionary dataLayerProperties;
- private IDictionary<string,dynamic> translatedProperties = null;
- protected virtual IDictionary<string,dynamic> Properties
- {
- if(translatedProperties == null)
- {
- translatedProperties = TranslateDictionary(dataLayerProperties);
- }
- return translatedProperties;
- }
- public IDictionary<string,dynamic> TranslateDictionary(IDictionary values)
- {
- return values.Keys.Cast<string>().ToDictionary(key=>key,key => values[key] as dynamic);
- }
现在,这种方法有明显的缺点吗?如果需要刷新dataLayerProperties怎么办?你必须将translateProperties设置为null等等.
选项2
这是我的首选方法.
- public class TranslatedDictionary : IDictionary<string,dynamic>
- {
- private IDictionary Original = null;
- public TranslatedDictionary(IDictionary original)
- {
- Original = original;
- }
- public ICollection<string> Keys
- {
- get
- {
- return Original.Keys.Cast<string>().ToList();
- }
- }
- public dynamic this[string key]
- {
- get
- {
- return Original[key] as dynamic;
- }
- set
- {
- Original[key] = value;
- }
- }
- // and so forth,for each method of IDictionary<string,dynamic>
- }
- //elsewhere,using your original property and field names:
- Properties = new TranslatedDictionary(properties);
现在,这个方法也有明显的缺点,最令人瞩目的是,Keys(和Value)和其他任何在IDictionary上返回ICollection的东西都必须为每个调用返回一个新的数组,但这仍然允许最灵活的方法,因为它确保数据始终是最新的.