我正在尝试使用反射来创建结构
这是我正在使用的代码:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Reflection;
public interface ISomeInterface
{
}
public struct SomeStruct : ISomeInterface
{
public int Value;
public SomeStruct(int value)
{
Value = value;
}
}
public struct Filter
{
public SomeStruct SomeStruct;
}
public class SomeClass
{
private List<ISomeInterface> someInterfaces = new List<ISomeInterface>();
public void Add(ISomeInterface someInterface)
{
this.someInterfaces.Add(someInterface);
}
public TFilter ToFilter<TFilter>() where TFilter : struct
{
var filterType = typeof(TFilter);
var filterFields = filterType.GetFields();
var result = Activator.CreateInstance<TFilter>();
foreach (var filterField in filterFields)
{
var component = this.someInterfaces.First(s => s.GetType() == filterField.FieldType);
filterField.SetValue(result, component);
}
return result;
}
}
class Program
{
public static void Main()
{
var someClass = new SomeClass();
var someStruct = new SomeStruct(10);
someClass.Add(someStruct);
var filter = someClass.ToFilter<Filter>();
// Should return 10, but returns 0, why?
Console.WriteLine(filter.SomeStruct.Value);
Console.ReadLine();
}
}
由于某种原因,ToFilter方法的结果返回一个具有不正确值的结构。我已经在这几个小时了,我只是不明白发生了什么。
几个小时以来我一直很沮丧,如果有人能帮助我,我会非常感激。
问题出在这一行:
filterField.SetValue(result, component);
SetValue
方法采用object
作为它的第一个参数。但result
是Filter
类型,这是一个struct
。这会导致result
被装箱,然后在SetValue
内部设置了盒装实例的字段。这对原来的result
没有影响,它的价值仍为零。
传递结构时应该小心,因为这样的意外行为总会发生。如果您不知道装箱/拆箱的工作方式以及值类型和引用类型之间的差异,则不应使用此类结构。
Here是一个很好的起点。