【问题标题】:Serializing classes in a dynamic assembly在动态程序集中序列化类
【发布时间】:2010-08-19 20:43:14
【问题描述】:

在运行时,我生成一个动态程序集,其中包含我的应用程序的数据模型。每个类都用DataContractAttribute 注释,每个属性都用DataMemberAttribute 注释。现在,如果我尝试序列化一个对象,生成的 XML 只包含根节点,但没有属性。但是,如果我在代码中定义完全相同的类 - 那么它工作正常。 到目前为止,我正在使用“用户”对象进行测试。这是“静态”实现:

[DataContract]
private class NonDynamicUser
{
    [DataMember]
    public Guid Id { get; set; }

    [DataMember]
    public String Username { get; set; }

    [DataMember]
    public String Password { get; set; }
}

这里是“动态”的:

AssemblyBuilder assemblyBuilder = AppDomain.CurrentDomain.DefineDynamicAssembly(new AssemblyName(assemblyName), AssemblyBuilderAccess.RunAndCollect);
ModuleBuilder moduleBuilder = assemblyBuilder.DefineDynamicModule("DynamicModule");
TypeBuilder userTypeBuilder = moduleBuilder.DefineType(assemblyName + "." + "User", TypeAttributes.Public);
Type attrType = typeof(DataContractAttribute);
userTypeBuilder.SetCustomAttribute(new CustomAttributeBuilder(attrType.GetConstructor(Type.EmptyTypes), new object[] { }));
CreateFieldForType(userTypeBuilder, typeof(Guid), "Id");
CreateFieldForType(userTypeBuilder, typeof(String), "Username");
CreateFieldForType(userTypeBuilder, typeof(String), "Password");
Type userType = userTypeBuilder.CreateType();

void CreateFieldForType(TypeBuilder typeBuilder, Type fieldType, String fieldName) 方法基本上是创建一个私有字段,然后是一个带有公共 getter 和 setter 的公共属性(用 DataMemberAttribute 注释)。

我通过反射再次创建要序列化的实例,这适用于“动态”对象以及“静态”对象:

object reflectedUser = Activator.CreateInstance(userType);
//object reflectedUser = Activator.CreateInstance(typeof(NonDynamicUser));
reflectedUser.GetType().GetProperty("Id").SetValue(reflectedUser, Guid.NewGuid(), null);
reflectedUser.GetType().GetProperty("Username").SetValue(reflectedUser, "s7orm", null);
reflectedUser.GetType().GetProperty("Password").SetValue(reflectedUser, "s7orm", null);

这就是我序列化对象的方式(直截了当的东西):

DataContractSerializer ser = new DataContractSerializer(reflectedUser.GetType());
ser.WriteObject(writer, reflectedUser);

这会产生不同的输出,具体取决于使用的是动态用户类还是静态用户类...

你知道我做错了什么吗?

更新: 这是CreateFieldForType方法的代码:

private static void CreateFieldForType(TypeBuilder typeBuilder, Type fieldType, String fieldName)
{
    Type attrType = typeof(DataMemberAttribute);
    CustomAttributeBuilder attr = new CustomAttributeBuilder(attrType.GetConstructor(Type.EmptyTypes), new object[] { });

    FieldBuilder fieldBuilder = typeBuilder.DefineField("_" + fieldName.ToLowerInvariant(), fieldType, FieldAttributes.Private);

    PropertyBuilder propertyBuilder = typeBuilder.DefineProperty(fieldName, PropertyAttributes.HasDefault, fieldType, null);
    propertyBuilder.SetCustomAttribute(attr);

    MethodAttributes getterAndSetterAttributes = MethodAttributes.Public | MethodAttributes.SpecialName | MethodAttributes.HideBySig | MethodAttributes.Virtual;

    MethodBuilder getMethodBuilder = typeBuilder.DefineMethod("get_" + propertyBuilder.Name, getterAndSetterAttributes, fieldType, Type.EmptyTypes);

    ILGenerator getMethodILGenerator = getMethodBuilder.GetILGenerator();
    getMethodILGenerator.Emit(OpCodes.Ldarg_0);
    getMethodILGenerator.Emit(OpCodes.Ldfld, fieldBuilder);
    getMethodILGenerator.Emit(OpCodes.Ret);

    MethodBuilder setMethodBuilder = typeBuilder.DefineMethod("set_" + propertyBuilder.Name, getterAndSetterAttributes, null, new Type[] { fieldType });

    ILGenerator setMethodILGenerator = setMethodBuilder.GetILGenerator();
    setMethodILGenerator.Emit(OpCodes.Ldarg_0);
    setMethodILGenerator.Emit(OpCodes.Ldarg_1);
    setMethodILGenerator.Emit(OpCodes.Stfld, fieldBuilder);
    setMethodILGenerator.Emit(OpCodes.Ret);

    propertyBuilder.SetGetMethod(getMethodBuilder);
    propertyBuilder.SetSetMethod(setMethodBuilder);
}

【问题讨论】:

  • 您可以使用 .NET 反射器查看您生成的代码与编译器为您的静态版本生成的代码之间的差异。
  • 我已经尝试过了,但是当我用红色门的反射器打开动态程序集(当然首先保存程序集)时,它没有显示任何类。一些奇怪的事情正在发生......
  • 为方便起见,您可以粘贴您的 CreateFieldForType 代码吗?
  • 没问题,我更新了帖子。

标签: c# serialization


【解决方案1】:

将模块名称更改为与程序集/dll 名称相同允许我在 Reflector 中看到该类:

ModuleBuilder moduleBuilder = assemblyBuilder.DefineDynamicModule("blah.dll");

我创建了自己的 CreateFieldForType 方法版本,它对我有用。这是我确定我正确生成类型的方式:

userInstance.GetType().GetProperty("Id").SetValue(userInstance, Guid.NewGuid(), null);
userInstance.GetType().GetProperty("Username").SetValue(userInstance, "Test1", null);
userInstance.GetType().GetProperty("Password").SetValue(userInstance, "Test2", null);

MemoryStream stream = new MemoryStream();
DataContractSerializer ser = new DataContractSerializer(userInstance.GetType());
ser.WriteObject(stream, userInstance);

stream.Seek(0, SeekOrigin.Begin);
object reSerialized = ser.ReadObject(stream);
foreach (var property in reSerialized.GetType().GetProperties(BindingFlags.Instance | BindingFlags.Public))
    Console.WriteLine("{0}: {1}", property.Name, property.GetValue(reSerialized, null));

我采用了您刚刚发布的方法并针对我的代码运行了它。它没有用。然后我慢慢地用我创建的代码替换它。我发现破坏您的代码(使我的验证码停止工作)的行是这一行:

MethodAttributes getterAndSetterAttributes = MethodAttributes.Public | MethodAttributes.SpecialName | MethodAttributes.HideBySig | MethodAttributes.Virtual;

当我删除 MethodAttributes.Virtual 标志后,代码的工作方式与 NonDynamicUser 相同。

我建议你让你生成的程序集正常工作,这样你就可以在反射器中看到生成的类。这将使验证进一步修改变得更加容易(例如,您可以添加 virtual 关键字)。

【讨论】:

  • 哇,我觉得自己很愚蠢 :) 我添加了虚拟标志来测试一些代理库,然后忘记删除它......非常感谢!而且你的动态模块命名技巧是无价的!​​span>
  • @s7orm:很高兴我能帮上忙。干杯!
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2016-11-02
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多