我正在尝试构建一个静态通用缓存,它将保存Apply(事件e)形式的几种方法的方法信息。来自CQRS Lokad框架的RedirectToWhen中使用了类似的代码。
问题是泛型缓存不考虑派生类(如果有的话)。这是一个简单的测试,显示了不良行为:
[TestMethod]
public void TestGenericsInheritance()
{
var sut = new DerivedFromAbstractType();
Utils.UsefulMethod<DerivedFromAbstractType>(sut);
Assert.AreEqual(10, sut.Value);
}
public abstract class AbstractType
{
public int Value { get; set; }
}
public class DerivedFromAbstractType : AbstractType
{
public void AnyOtherMethod()
{
Value = 10;
}
}
public static class Utils
{
public static void UsefulMethod<T>(T instance)
{
MethodInfo info = typeof(T)
.GetMethods(BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.Instance)
.Where(m => m.Name == "AnyOtherMethod")
.Where(m => m.GetParameters().Length == 0).FirstOrDefault();
info.Invoke(instance,null);
}
}
typeof(T)返回AbstractType,因此我无法使用它来构建通用的静态缓存。 如何获得知道派生类型的方法的通用缓存?
答案 0 :(得分:0)
使用
instance.GetType()
如果instance不为null且typeof(T)os实例为null。
答案 1 :(得分:0)
我知道这有点恶意,但我有同样的问题,typeof(T)返回基类型。当想要创建实现Emit / Apply模式的BaseAggregate时,这很有用。我这样解决了:
private static class Cache
{
private static readonly IDictionary<Type, IDictionary<Type, MethodInfo>> _dict =
new Dictionary<Type, IDictionary<Type, MethodInfo>>();
public static IDictionary<Type, MethodInfo> GetDictionaryForType(Type type)
{
if (_dict.ContainsKey(type))
{
return _dict[type];
}
var dict = type.GetMethods(BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.Instance)
.Where(m => m.Name == "When")
.Where(m => m.GetParameters().Length == 1)
.ToDictionary(m => m.GetParameters().First().ParameterType, m => m);
_dict.Add(type, dict);
return dict;
}
}
这仍然可以很好地缓存,object.GetType()
返回正确的类型以输入此方法。