使用反射访问对象中的Struct的字段

时间:2012-11-09 12:22:37

标签: c# reflection typedreference

我正在尝试使用包含object的反射(最终在编译时未知)struct。我已经到了TypedReference.MakeTypedReference,但我已经撞墙了。

这是我的班级和结构

public class MyObject
{
    public int Id;
    public Money Amount; 
}

public struct Money
{
    public int Vaule;
    public string Code;
}

以下是我尝试使用反射在MyObject中设置“Amount”的“代码”。正如我上面提到的,我正在寻找一种在编译时不了解这些类型的解决方案(这太容易了!)

这是我到目前为止的代码(我使用[0],[1]来使代码更简单)

var obj = new MyObject() { Id = 1 };
obj.Amount.Vaule = 10;
obj.Amount.Code = "ABC";

FieldInfo[] objFields = obj.GetType().GetFields();
FieldInfo[] moneyFields = objFields[1].GetValue(obj).GetType().GetFields();

List<FieldInfo> fields = new List<FieldInfo>() { objFields[1] };
fields.AddRange( moneyFields );

TypedReference typeRef = TypedReference.MakeTypedReference( 
                           objFields[1].GetValue( obj ), fields.ToArray() );

moneyFields[1].SetValueDirect( typeRef, "XXX" );

TypedReference.MakeTypedReference爆炸了; “FieldInfo与目标类型不匹配。”同样,如果我只是通过objFields[1]。如果只传递moneyFields我得到“TypedReferences不能重新定义为原语。”

为什么?假设我正在创建随机测试夹具,并希望使用随机数据填充类字段:)

1 个答案:

答案 0 :(得分:17)

坦率地说,TypedReference这里没有任何必要 - 只需一个盒装结构就可以正常工作:

    var amountField = obj.GetType().GetField("Amount");
    object money = amountField.GetValue(obj);
    var codeField = money.GetType().GetField("Code");
    codeField.SetValue(money, "XXX");
    amountField.SetValue(obj, money);

然而!我会告诉你一些事情:

  • 公共领域而不是属性通常不是一个好主意;以后经常会咬你的话
  • 可变结构(即可以在创建后更改的结构)几乎从不一个好主意,并且会更频繁地咬,并且更难咬
  • 将可变结构和公共字段组合在一起,但后来更改
  • 会使问题变得很困难