在 C# 中有没有一种方法可以使用反射来设置对象属性?
前任:
MyObject obj = new MyObject();
obj.Name = "Value";
我想用反射设置 obj.Name
。就像是:
Reflection.SetProperty(obj, "Name") = "Value";
有没有办法做到这一点?
是的,您可以使用 Type.InvokeMember()
:
using System.Reflection;
MyObject obj = new MyObject();
obj.GetType().InvokeMember("Name",
BindingFlags.Instance | BindingFlags.Public | BindingFlags.SetProperty,
Type.DefaultBinder, obj, "Value");
如果 obj
没有名为 Name
的属性,或者无法设置,这将引发异常。
另一种方法是获取属性的元数据,然后设置它。这将允许您检查属性是否存在,并验证它是否可以设置:
using System.Reflection;
MyObject obj = new MyObject();
PropertyInfo prop = obj.GetType().GetProperty("Name", BindingFlags.Public | BindingFlags.Instance);
if(null != prop && prop.CanWrite)
{
prop.SetValue(obj, "Value", null);
}
你也可以这样做:
Type type = target.GetType();
PropertyInfo prop = type.GetProperty("propertyName");
prop.SetValue (target, propertyValue, null);
其中 target 是将设置其属性的对象。
反射,基本上,即
myObject.GetType().GetProperty(property).SetValue(myObject, "Bob", null);
或者有一些库可以在便利性和性能方面提供帮助;例如 FastMember:
var wrapped = ObjectAccessor.Create(obj);
wrapped[property] = "Bob";
(这还有一个好处是不需要提前知道是字段还是属性)
或者你可以在你自己的扩展类中包装 Marc 的一个衬垫:
public static class PropertyExtension{
public static void SetPropertyValue(this object obj, string propName, object value)
{
obj.GetType().GetProperty(propName).SetValue(obj, value, null);
}
}
并这样称呼它:
myObject.SetPropertyValue("myProperty", "myValue");
为了更好地衡量,让我们添加一个方法来获取属性值:
public static object GetPropertyValue(this object obj, string propName)
{
return obj.GetType().GetProperty(propName).GetValue (obj, null);
}
是的,使用 System.Reflection
:
using System.Reflection;
...
string prop = "name";
PropertyInfo pi = myObject.GetType().GetProperty(prop);
pi.SetValue(myObject, "Bob", null);
使用这样的东西:
public static class PropertyExtension{
public static void SetPropertyValue(this object p_object, string p_propertyName, object value)
{
PropertyInfo property = p_object.GetType().GetProperty(p_propertyName);
property.SetValue(p_object, Convert.ChangeType(value, property.PropertyType), null);
}
}
或者
public static class PropertyExtension{
public static void SetPropertyValue(this object p_object, string p_propertyName, object value)
{
PropertyInfo property = p_object.GetType().GetProperty(p_propertyName);
Type t = Nullable.GetUnderlyingType(property.PropertyType) ?? property.PropertyType;
object safeValue = (value == null) ? null : Convert.ChangeType(value, t);
property.SetValue(p_object, safeValue, null);
}
}
您还可以使用类似的方式访问字段:
var obj=new MyObject();
FieldInfo fi = obj.GetType().
GetField("Name", BindingFlags.NonPublic | BindingFlags.Instance);
fi.SetValue(obj,value)
通过反射,一切都可以成为一本打开的书:) 在我的示例中,我们绑定到一个私有实例级字段。
当您想使用属性名称从另一个对象批量分配一个对象的属性时,您可以试试这个:
public static void Assign(this object destination, object source)
{
if (destination is IEnumerable && source is IEnumerable)
{
var dest_enumerator = (destination as IEnumerable).GetEnumerator();
var src_enumerator = (source as IEnumerable).GetEnumerator();
while (dest_enumerator.MoveNext() && src_enumerator.MoveNext())
dest_enumerator.Current.Assign(src_enumerator.Current);
}
else
{
var destProperties = destination.GetType().GetProperties();
foreach (var sourceProperty in source.GetType().GetProperties())
{
foreach (var destProperty in destProperties)
{
if (destProperty.Name == sourceProperty.Name && destProperty.PropertyType.IsAssignableFrom(sourceProperty.PropertyType))
{
destProperty.SetValue(destination, sourceProperty.GetValue(source, new object[] { }), new object[] { });
break;
}
}
}
}
我刚刚发布了一个 Nuget 包,它不仅可以设置第一级属性,还可以设置给定对象中任意深度的嵌套属性。
这里是the package
通过从根开始的路径设置对象的属性值。
对象可以是复杂对象,属性可以是多级深层嵌套属性,也可以是根目录下的属性。 ObjectWriter
将使用属性路径参数查找属性并更新其值。属性路径是从根到我们要设置的结束节点属性访问的属性的附加名称,由分隔符字符串参数分隔。
用法:
要直接在对象根下设置属性:
IE。 LineItem
类有一个名为 ItemId
的 int 属性
LineItem lineItem = new LineItem();
ObjectWriter.Set(lineItem, "ItemId", 13, delimiter: null);
用于在对象根下设置多个级别的嵌套属性:
IE。 Invite
类有一个名为 State
的属性,它有一个名为 Invite
的属性(Invite 类型),它有一个名为 Recipient
的属性,它有一个名为 Id
的属性。
更复杂的是,State
属性不是引用类型,而是 struct
。
以下是如何在单行中设置对象树底部的 Id 属性(设置为“outlook”的字符串值)。
Invite invite = new Invite();
ObjectWriter.Set(invite, "State_Invite_Recipient_Id", "outlook", delimiter: "_");
根据 MarcGravell 的建议,我构造了以下静态方法。该方法通常使用 FastMember 将源对象中的所有匹配属性分配给目标
public static void DynamicPropertySet(object source, object target)
{
//SOURCE
var src_accessor = TypeAccessor.Create(source.GetType());
if (src_accessor == null)
{
throw new ApplicationException("Could not create accessor!");
}
var src_members = src_accessor.GetMembers();
if (src_members == null)
{
throw new ApplicationException("Could not fetch members!");
}
var src_class_members = src_members.Where(x => x.Type.IsClass && !x.Type.IsPrimitive);
var src_class_propNames = src_class_members.Select(x => x.Name);
var src_propNames = src_members.Except(src_class_members).Select(x => x.Name);
//TARGET
var trg_accessor = TypeAccessor.Create(target.GetType());
if (trg_accessor == null)
{
throw new ApplicationException("Could not create accessor!");
}
var trg_members = trg_accessor.GetMembers();
if (trg_members == null)
{
throw new ApplicationException("Could not create accessor!");
}
var trg_class_members = trg_members.Where(x => x.Type.IsClass && !x.Type.IsPrimitive);
var trg_class_propNames = trg_class_members.Select(x => x.Name);
var trg_propNames = trg_members.Except(trg_class_members).Select(x => x.Name);
var class_propNames = trg_class_propNames.Intersect(src_class_propNames);
var propNames = trg_propNames.Intersect(src_propNames);
foreach (var propName in propNames)
{
trg_accessor[target, propName] = src_accessor[source, propName];
}
foreach (var member in class_propNames)
{
var src = src_accessor[source, member];
var trg = trg_accessor[target, member];
if (src != null && trg != null)
{
DynamicPropertySet(src, trg);
}
}
}
var val = Convert.ChangeType(propValue, propInfo.PropertyType);
来源:devx.com/vb2themax/Tip/19599obj.GetType().GetProperty("Name")?.GetSetMethod()?.Invoke(...)
CanWrite=False
类型设置值,对吧?