- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我有许多数据模型对象,它们的主键字段用 DataAnnotations.KeyAttributes 装饰:
public class Example
{
[Key]
[Column("FirstKey", Order = 1)]
public Guid FirstKey { get; set; }
[Key]
[Column("SecondKey", Order = 2)]
public Guid SecondKey { get; set; }
[Column("Unimportant", Order = 3)]
public int UnimportantField {get; set; }
}
我可以使用 PropertInfo 选择数据模型对象的关键字段:
public static IEnumerable<PropertyInfo> GetKeyProperties<T>(this T source)
{
PropertyInfo[] properties = source.GetType().GetProperties();
return properties.Where(p => Attribute.IsDefined(p, typeof(KeyAttribute)));
}
我还可以手动构建一个包含给定类的键字段的匿名对象:
public static object GetKeyObjectForExample(Example example)
{
return new { example.FirstKey, example.SecondKey };
}
我想创建一个通用方法来构造一个 Func 委托(delegate),该委托(delegate)将返回一个包含键字段的匿名对象。
public static Func<T, object> GetKeyFunction<T>(this T source)
{
//construct Func<T, object> using IEnumerable<PropertyInfo>?
}
因此,为给定的 Example 实例调用 GetKeyFunction 方法将返回一个 Func 委托(delegate)。此委托(delegate)应生成与 GetKeyObjectForExample 方法等效的输出。
如何实现 GetKeyFunction 方法?
最佳答案
不确定您要实现的确切目标,因为即使使用动态或匿名对象,您仍然需要知道关键字段的名称,但像这样的方法可以解决您声明的方法(错误处理等省略):
using System;
using System.Collections.Generic;
using System.Linq;
using System.Reflection;
using System.Dynamic;
namespace ConsoleApplication11
{
class KeyAttribute : Attribute
{
}
public class Example
{
[Key]
public Guid FirstKey { get; set; }
[Key]
public Guid SecondKey { get; set; }
public int UnimportantField { get; set; }
}
static class Program
{
static void Main(string[] args)
{
Example e = new Example()
{
FirstKey = Guid.NewGuid(),
SecondKey = Guid.NewGuid()
};
var func = GetKeyFunction<Example>();
var key = func(e);
Console.WriteLine(key.FirstKey);
Console.WriteLine(key.SecondKey);
}
public static Func<T, dynamic> GetKeyFunction<T>()
{
return (t) =>
{
var dyn = (IDictionary<string, object>)new ExpandoObject();
foreach (var p in GetKeyProperties(typeof(T)))
{
dyn.Add(p.Name, p.GetValue(t));
}
return dyn;
};
}
public static IEnumerable<PropertyInfo> GetKeyProperties(Type t)
{
PropertyInfo[] properties = t.GetProperties();
return properties.Where(p => p.GetCustomAttributes<KeyAttribute>().Any());
}
}
}
编辑
阅读一些评论后,您可能会喜欢上这种方法,并向您的动态对象添加一个 Equals 和 GetHashCode 实现(此实现假定关键字段的名称在相等性方面很重要)。感觉像确定对象相等性的代码和动态太多了,但是......
与上面相同的代码,但有一个辅助类来确定字典相等性和哈希码。
using System;
using System.Collections.Generic;
using System.Linq;
using System.Reflection;
using System.Dynamic;
namespace ConsoleApplication11
{
class KeyAttribute : Attribute
{
}
public class Example
{
[Key]
public Guid FirstKey { get; set; }
[Key]
public Guid SecondKey { get; set; }
public int UnimportantField { get; set; }
}
static class Program
{
static void Main(string[] args)
{
Example e = new Example()
{
FirstKey = Guid.NewGuid(),
SecondKey = Guid.NewGuid(),
UnimportantField = 0
};
var func = GetKeyFunction<Example>();
var key = func(e);
Console.WriteLine(key.FirstKey);
Console.WriteLine(key.SecondKey);
Console.WriteLine(key.Equals(key));
Console.WriteLine(key.GetHashCode());
Example e2 = new Example()
{
FirstKey = Guid.NewGuid(),
SecondKey = Guid.NewGuid(),
UnimportantField = 1
};
var key2 = func(e2);
Console.WriteLine(key2.FirstKey);
Console.WriteLine(key2.SecondKey);
Console.WriteLine(key.Equals(key2));
Console.WriteLine(key2.GetHashCode());
}
public static Func<T, dynamic> GetKeyFunction<T>()
{
return (t) =>
{
var dyn = (IDictionary<string, object>)new ExpandoObject();
foreach (var p in GetKeyProperties(typeof(T)))
{
dyn.Add(p.Name, p.GetValue(t));
}
dyn.Add("Equals", (Func<object, bool>)((o) =>
{
var comparer = new DictionaryComparer<string,object>();
return comparer.Equals(dyn as IDictionary<string,object>, o as IDictionary<string,object>);
}
));
dyn.Add("GetHashCode", (Func<int>)(() =>
{
var comparer = new DictionaryComparer<string, object>();
return comparer.GetHashCode(dyn as IDictionary<string, object>);
}
));
return dyn;
};
}
public static IEnumerable<PropertyInfo> GetKeyProperties(Type t)
{
PropertyInfo[] properties = t.GetProperties();
return properties.Where(p => p.GetCustomAttributes<KeyAttribute>().Any());
}
}
public class DictionaryComparer<TKey, TValue> : EqualityComparer<IDictionary<TKey, TValue>>
{
public DictionaryComparer()
{
}
public override bool Equals(IDictionary<TKey, TValue> x, IDictionary<TKey, TValue> y)
{
// early-exit checks
if (object.ReferenceEquals(x, y))
return true;
if (null == x || y == null)
return false;
if (x.Count != y.Count)
return false;
// check keys are the same
foreach (TKey k in x.Keys)
if (!y.ContainsKey(k))
return false;
// check values are the same
foreach (TKey k in x.Keys)
{
TValue v = x[k];
if (object.ReferenceEquals(v, null))
return object.ReferenceEquals(y[k], null);
if (!v.Equals(y[k]))
return false;
}
return true;
}
public override int GetHashCode(IDictionary<TKey, TValue> obj)
{
if (obj == null)
return 0;
int hash = 0;
foreach (KeyValuePair<TKey, TValue> pair in obj)
{
int key = pair.Key.GetHashCode(); // key cannot be null
int value = pair.Value != null ? pair.Value.GetHashCode() : 0;
hash ^= ShiftAndWrap(key, 2) ^ value;
}
return hash;
}
private static int ShiftAndWrap(int value, int positions)
{
positions = positions & 0x1F;
// Save the existing bit pattern, but interpret it as an unsigned integer.
uint number = BitConverter.ToUInt32(BitConverter.GetBytes(value), 0);
// Preserve the bits to be discarded.
uint wrapped = number >> (32 - positions);
// Shift and wrap the discarded bits.
return BitConverter.ToInt32(BitConverter.GetBytes((number << positions) | wrapped), 0);
}
}
}
关于c# - 将 PropertyInfo 转换为 Func<> 委托(delegate),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/23159957/
我正在制作一个辅助方法,它会自动为给定实体(类)的属性设置随机值,这样我就不必在测试时为每个属性填充一个值。 在我的例子中,每个实体都继承自 BaseEntity 类,该类具有 ID、CreatedB
我知道 typeof(T) == typeof(T)永远为真,因为 Type对象基本上是静态的,只有一个 Type每个类类型都存在实例(如果这是错误的,请纠正我......我有几个程序在这个假设下运行
我有以下类属性: [EffectAspect(Enums.Effects.Low)] public int Wind { set; get; } [EffectAspect(Enums.Effects
例如,我有这个简单的类: public class MyClass { public String MyProperty { get; set; } } 获取 MyProperty 的 Prope
假设我有这门课: class Test123 where T : struct { public Nullable Test {get;set;} } 和这个类 class Test321 {
如何按照类中的排列顺序获取类型 FieldInfos/PropertyInfos 作为 MemberInfo 数组? class Test { public bool First { get;
我想获取特定属性的 PropertyInfo。我可以使用: foreach(PropertyInfo p in typeof(MyObject).GetProperties()) { if (
我想忽略属性信息如下的属性; PropertyInfo propertyInfo = typeof(GLAccount).GetProperty("ExampleProp"); modelBuilde
我验证了 GetHashCode 和 Equals 允许将 PropertyInfo 用作字典键。 (具体来说,哈希码相同,Equals 返回 true。) 那么使用 PropertyInfo 作为字
我对 C# 反射有疑问。我要反射(reflect)的对象如下: public partial class ApplicationUser : IdentityUser { public App
我在反射、动态调用对象和读取集合值方面遇到问题。 在 Referenced COM/Interop 中,它看起来像这样: ICollection collection = (ICollection)s
这个问题在这里已经有了答案: PropertyInfo : is the property an indexer? (2 个答案) 关闭 7 年前。 我在一个类中有一个索引器方法,允许我这样做: v
这个问题在这里已经有了答案: Using PropertyInfo.GetValue() (1 个回答) 关闭 5 年前。 我有以下类(class): public class MagicMetad
我需要通过 Tag 类在 BFrame 类中设置 Value 属性。 我应该如何设置 Value 属性(property)? 澄清: 我不是要设置 Frame 的值 内的属性 Tag 类但是 Valu
使用反射我只想检索具有 get 和 set 方法的属性,而忽略只有 get 的属性.我想做的是为用户提供他/她能够更改的变量列表,因此向他们显示只有 get 方法的属性会产生误导。 给定下面的代码,用
这个问题在这里已经有了答案: How to get the PropertyInfo of a specific property? (5 个答案) 关闭 9 年前。 我在 Myclass 中有一个
我正在创建一个小型验证框架,我有一个可分配给方法的自定义 Validation Attribute 和 ValidationCore 中的一个 IsValid 属性> 类。当 IsValid 在方法内
在第一个内部循环中,当从 PropertyInfo[] 获取值时,我能够传递正确的目标对象,但是在第二个内部循环中,它给出了目标对象不正确的异常. 所以我想要的是获取具有此 listPropertie
我有一个带有子类地址的客户类 internal class Customer { public int id { get; set; } public string name {
我遇到了与 this one 类似的错误,但不幸的是不是同样简单的解决方案。这是代码: public virtual void MapObject(T obj, IViewModel viewMode
我是一名优秀的程序员,十分优秀!