Ari*_*oth 3 c# mapping reflection
尽管查看了多个 SO 帖子和我能想到的任何其他内容,但我在这里完全不知所措。
我的目标是制作一个非常非常简单的映射器。我基本上可以在某些单元测试中用作工具的东西。它不需要很复杂或任何东西——只需将一个对象的高级原始值和字符串值映射到另一个对象。所以基本算法是:
TFromTToTTo我们可以复制到的实例。from对象的值to对象上设置值问题是,无论我做什么,无论属性的类型是什么(int或者string,例如),我都会得到以下信息:
对象与目标类型不匹配。
这是我正在使用的代码:
public TTo Map<TFrom, TTo>(TFrom from)
{
if (from == null) return default;
var fromProps = GetProperties(typeof(TFrom));
var toProps = GetProperties(typeof(TTo));
// Props that can be mapped from one to the other
var propsToCopy = fromProps.Intersect(toProps, new PropertyComparer()).ToList();
var returnObject = (TTo)Activator.CreateInstance(typeof(TTo));
foreach (var prop in propsToCopy)
{
// Copy the values
var fromValue = prop.GetValue(from, null);
var convertedValue = Convert.ChangeType(fromValue, prop.PropertyType);
prop.SetValue(returnObject, convertedValue, null);
}
return returnObject;
}
public PropertyInfo[] GetProperties(Type objectType)
{
var allProps = objectType.GetProperties(
BindingFlags.Public | BindingFlags.Instance);
return allProps.Where(p => p.PropertyType.IsPrimitive ||
p.PropertyType == typeof(string)).ToArray();
}
private class PropertyComparer : IEqualityComparer<PropertyInfo>
{
public bool Equals(PropertyInfo x, PropertyInfo y)
{
return x.Name.Equals(y.Name);
}
public int GetHashCode(PropertyInfo obj)
{
return obj.Name.GetHashCode();
}
}
Run Code Online (Sandbox Code Playgroud)
这是我将其称为示例类的示例:
public class Foo
{
public string StringProp { get; set; }
public int IntProp { get; set; }
}
public class FooOther
{
public string StringProp { get; set; }
public int IntProp { get; set; }
}
var foo = new Foo { IntProp = 1, StringProp = "foo" };
var mappedFoo = Map<Foo, FooOther>(foo);
Run Code Online (Sandbox Code Playgroud)
我从 Visual Studio 中得到的唯一提示来自监视窗口:如果属性类型是 a string,监视窗口报告的类型为convertedValueas object。如果属性类型是int,监视窗口会报告object {int}。
您正在使用的 PropertyInfo 仍然耦合到它所表示的属性所属的类型,因此您无法使用它来设置另一种类型的对象的值,而不会出现错误。
这是行为的一个简短示例:
public class A {
public string Id {get;set;}
}
public class B {
public string Id {get;set;}
}
void Main()
{
var test = new A() { Id = "Test"};
var prop = test.GetType().GetProperty("Id");
var b = (B)Activator.CreateInstance(typeof(B));
var fromValue = prop.GetValue(test);
var converted = Convert.ChangeType(fromValue, prop.PropertyType);
prop.SetValue(b, converted, null); // Exception
}
Run Code Online (Sandbox Code Playgroud)
如果您将 PropertyInfo 视为 A 的成员,这是有道理的。要解决此问题,您需要获取特定于您的类型的属性信息。我可以使用以下内容修复我的示例:
var propTo = typeof(B).GetProperty(prop.Name);
propTo.SetValue(b, converted, null);
Console.WriteLine(b.Id); // Output: Test
Run Code Online (Sandbox Code Playgroud)
综上所述,如果您将 foreach 的内容更改为以下内容,您应该清楚:
foreach (var prop in propsToCopy)
{
// Copy the values
var fromValue = prop.GetValue(from, null);
var convertedValue = Convert.ChangeType(fromValue, prop.PropertyType);
var propTo = typeof(TTO).GetProperty(prop.Name);
propTo.SetValue(returnObject, convertedValue, null);
}
Run Code Online (Sandbox Code Playgroud)