//Get PropertyDescriptor object for the given property name
var propDesc = TypeDescriptor.GetProperties(typeof(T))[propName];
//Get FillAttributes methodinfo delegate
var methodInfo = propDesc.GetType().GetMethods(BindingFlags.Instance | BindingFlags.Public |
BindingFlags.NonPublic)
.FirstOrDefault(m => m.IsFamily || m.IsPublic && m.Name == "FillAttributes");
//Create Validation attribute
var attribute = new RequiredAttribute();
var attributes= new ValidationAttribute[]{attribute};
//Invoke FillAttribute method
methodInfo.Invoke(propDesc, new object[] { attributes });
您好我正在尝试使用上面的代码在运行时添加Validation属性。但是我得到以下例外:
收藏是固定大小的
答案 0 :(得分:125)
不要让别人告诉你你不能这样做。如果你愿意,你可以竞选总统:-)
为了您的方便,这是一个完全有效的例子
public class SomeAttribute : Attribute
{
public SomeAttribute(string value)
{
this.Value = value;
}
public string Value { get; set; }
}
public class SomeClass
{
public string Value = "Test";
}
[TestMethod]
public void CanAddAttribute()
{
var type = typeof(SomeClass);
var aName = new System.Reflection.AssemblyName("SomeNamespace");
var ab = AppDomain.CurrentDomain.DefineDynamicAssembly(aName, AssemblyBuilderAccess.Run);
var mb = ab.DefineDynamicModule(aName.Name);
var tb = mb.DefineType(type.Name + "Proxy", System.Reflection.TypeAttributes.Public, type);
var attrCtorParams = new Type[] { typeof(string) };
var attrCtorInfo = typeof(SomeAttribute).GetConstructor(attrCtorParams);
var attrBuilder = new CustomAttributeBuilder(attrCtorInfo, new object[] { "Some Value" });
tb.SetCustomAttribute(attrBuilder);
var newType = tb.CreateType();
var instance = (SomeClass)Activator.CreateInstance(newType);
Assert.AreEqual("Test", instance.Value);
var attr = (SomeAttribute)instance.GetType()
.GetCustomAttributes(typeof(SomeAttribute), false)
.SingleOrDefault();
Assert.IsNotNull(attr);
Assert.AreEqual(attr.Value, "Some Value");
}
答案 1 :(得分:3)
public class test{
public string Name{ get; set; }
}
var prop = now DeepCloner.GetFastDeepClonerProperties(typeof(test)).First();
prop.Attributes.Add(new JsonIgnoreAttribute());
// now test and se if exist
prop = now DeepCloner.GetFastDeepClonerProperties(typeof(test)).First();
bool containAttr = prop.ContainAttribute<JsonIgnoreAttribute>()
// or
JsonIgnoreAttribute myAttr = prop.GetCustomAttribute<JsonIgnoreAttribute>();
答案 2 :(得分:0)
无法在运行时添加属性。属性是静态的,无法添加或删除。
类似的问题:
答案 3 :(得分:0)
它不是因为FillAttributes
方法需要IList类型的参数而你传递数组。以下是MemberDescriptor.FillAttributes的实施:
protected virtual void FillAttributes(IList attributeList) {
if (originalAttributes != null) {
foreach (Attribute attr in originalAttributes) {
attributeList.Add(attr);
}
}
}
正如您所看到的,FillAttributes
只会使用您所有属性的属性填充attributeList
参数。要使代码正常工作,请更改var attributes= new ValidationAttribute[]{attribute};
行:
var attributes = new ArrayList { attribute };
此代码与在运行时向类型属性添加属性无关。这是“从类型中提取的PropertyDescriptor
添加属性”,除非您尝试在运行时构建基于已存在类型的类型,否则没有任何意义。
答案 4 :(得分:0)
对不起,我参加聚会很晚。这是给那些以后可能会来这里的人的。最好的答案是太棒了。最近,开发了一个库,该库将所有复杂性抽象化了,这就像这样简单:
var attributeType = typeof(CustomAAttribute);
var attributeParams = new object[] { "Jon Snow" };
var typeExtender = new TypeExtender("ClassA");
typeExtender.AddProperty("IsAdded", typeof(bool), attributeType, attributeParams);
与之合作。有关如何安装和使用该库的详细信息,请参见here