我遇到了一个小问题,这个问题通常无法解决看似相同的问题。
考虑:
我有一组动态生成的类,继承自已知的基类(让我们称之为BaseClass
)。
这些动态生成的类还具有动态生成的Properties
及其相关属性。
属性也是自定义类,但不是动态生成的:
[AttributeUsage(AttributeTargets.Property)]
class TypeAttribute: Attribute
{
private Type _type;
public Type Type
{
get { return _type; }
}
public TypeAttribute(Type t)
{
_type = t;
}
}
然后我想,运行时当然要获取这个赋值属性的值:
List<PropertyInfo> result = target.GetType()
.GetProperties()
.Where(
p =>
p.GetCustomAttributes(typeof(TypeAttribute), true)
//.Where(ca => ((TypeAttribute)ca).)
.Any()
)
.ToList();
其中target
是BaseClass
的子类。但是列表result
是空的,这让我感到困惑。
我使用
添加属性PropertyBuilder propertyBuilder = tb.DefineProperty(propertyName,
PropertyAttributes.HasDefault, propertyType, null);
ConstructorInfo classCtorInfo = typeof(TypeAttribute).
GetConstructor(new Type[] { typeof(Type) });
CustomAttributeBuilder myCABuilder = new CustomAttributeBuilder(
classCtorInfo, new object[] { getType(dataType) });
propertyBuilder.SetCustomAttribute(myCABuilder);
其中dataType
是要存储在属性中的类型,tb
是该类的TypeBuilder
。
如果我在属性上getCustomAttributes()
,我会得到除我正在寻找的属性之外的预期属性。但是,如果我getCustomAttributesData()
我得到了所有这些,但我正在寻找的是CustomAttributeData
类型,并且不能转换为TypeAttribute
(如果我检查VS中的实例)调试器我可以看到包含的信息是TypeAttribute
)。
我猜这是问题的症状,但我找不到原因 - 更不用说解决方案了。
有人可以向我指出为什么result
列表是空的吗?
答案 0 :(得分:1)
适合我;你确定你的财产和getter或setter方法足够存在,它实际上显示为一个属性?
using System;
using System.Collections.Generic;
using System.Linq;
using System.Reflection;
using System.Reflection.Emit;
[AttributeUsage(AttributeTargets.Property)]
public class TypeAttribute : Attribute
{
private Type _type;
public Type Type
{
get { return _type; }
}
public TypeAttribute(Type t)
{
_type = t;
}
}
public class BaseClass
{
}
static class Program
{
static void Main()
{
var aName = new AssemblyName("MyAssembly");
var ab = AppDomain.CurrentDomain.DefineDynamicAssembly(
aName, AssemblyBuilderAccess.RunAndSave);
var mb = ab.DefineDynamicModule(aName.Name, aName.Name + ".dll");
var tb = mb.DefineType("MyType", TypeAttributes.Public, typeof(BaseClass));
var propertyName = "MyProperty";
var propertyType = typeof(int);
var propertyBuilder = tb.DefineProperty(propertyName,
PropertyAttributes.HasDefault, propertyType, null);
var classCtorInfo = typeof(TypeAttribute).
GetConstructor(new Type[] { typeof(Type) });
Type tArg = typeof(float); // for no real reason
var myCABuilder = new CustomAttributeBuilder(
classCtorInfo, new object[] { tArg });
propertyBuilder.SetCustomAttribute(myCABuilder);
var field = tb.DefineField("myField", propertyType, FieldAttributes.Private);
var getter = tb.DefineMethod("get_" + propertyName,
MethodAttributes.HideBySig | MethodAttributes.SpecialName | MethodAttributes.Public,
propertyType, Type.EmptyTypes);
propertyBuilder.SetGetMethod(getter);
var il = getter.GetILGenerator();
il.Emit(OpCodes.Ldarg_0);
il.Emit(OpCodes.Ldfld, field);
il.Emit(OpCodes.Ret);
var setter = tb.DefineMethod("set_" + propertyName,
MethodAttributes.HideBySig | MethodAttributes.SpecialName | MethodAttributes.Public,
typeof(void), new[] { typeof(int) });
il = setter.GetILGenerator();
il.Emit(OpCodes.Ldarg_0);
il.Emit(OpCodes.Ldarg_1);
il.Emit(OpCodes.Stfld, field);
il.Emit(OpCodes.Ret);
propertyBuilder.SetSetMethod(setter);
var target = Activator.CreateInstance(tb.CreateType());
List<PropertyInfo> result = target.GetType()
.GetProperties()
.Where(
p =>
p.GetCustomAttributes(typeof(TypeAttribute), true)
//.Where(ca => ((TypeAttribute)ca).)
.Any()
).ToList();
}
}
此外,如果您只关心它退出,使用Attribute.IsDefined
会更有效。
List<PropertyInfo> result = target.GetType().GetProperties()
.Where(p => Attribute.IsDefined(p, typeof(TypeAttribute), true)).ToList();
另请注意,我必须欺骗getType(dataType)
的东西 - 我无法看到它在做什么,因为它不在问题中。确保它返回Type
。