使用反射时,为什么值设置不正确?

时间:2018-09-10 20:10:54

标签: c# reflection system.reflection

我正在尝试使用反射来创建结构

这是我使用的代码:

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方法的结果返回的结构体值不正确。我已经呆了几个小时,只是不知道发生了什么。

好几个小时以来,我一直感到沮丧,如果有人可以帮助我,将不胜感激。

1 个答案:

答案 0 :(得分:0)

问题出在这一行:

filterField.SetValue(result, component);

SetValue方法以object作为第一个参数。但是resultFilter类型的struct。这将导致result被装箱,然后在SetValue内设置装箱实例的字段。该值对原始result没有影响,其值仍为零。

在传递结构时,请务必小心,因为这样的意外行为总是会发生。如果您不了解装箱/拆箱的工作方式以及值类型和引用类型之间的差异,则不应使用这样的结构。

Here是一个很好的起点。