我有许多数据模型对象,其主键字段用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选择数据模型对象的Key字段:
public static IEnumerable<PropertyInfo> GetKeyProperties<T>(this T source)
{
PropertyInfo[] properties = source.GetType().GetProperties();
return properties.Where(p => Attribute.IsDefined(p, typeof(KeyAttribute)));
}
我还可以手动构建一个包含给定类的Key字段的匿名对象:
public static object GetKeyObjectForExample(Example example)
{
return new { example.FirstKey, example.SecondKey };
}
我想创建一个泛型方法,它将构造一个Func委托,它将返回一个包含Key字段的匿名对象。
public static Func<T, object> GetKeyFunction<T>(this T source)
{
//construct Func<T, object> using IEnumerable<PropertyInfo>?
}
因此,为给定的Example实例调用GetKeyFunction方法将返回一个Func委托。该委托应该为GetKeyObjectForExample方法生成等效的输出。
如何实现GetKeyFunction方法?
答案 0 :(得分:0)
不确定你想要实现什么,即使使用动态或匿名对象,你仍然需要知道Key字段的名称,但是像这样的东西可以解决你所陈述的方法(错误处理等等)消隐):
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);
}
}
}