在Web API中序列化动态类型

时间:2012-11-30 05:08:19

标签: c# reflection asp.net-web-api

我正在尝试创建一个泛型函数,当给定Enum Type时,它将返回一个对象,当WebApi序列化时,它将提供外观漂亮的XML / Json输出。

当序列化为JSON时,此方法可以正常工作,但我无法使用XML。如果我使用XmlSerializer或DataContractSerializer手动序列化返回的对象,我会得到预期的结果。当WebApi本身试图从HttpRequest序列化它时,我得到如下错误:

  

System.Runtime.Serialization.SerializationException

     

使用数据合约名称键入“优先级”   “优先级:http://schemas.datacontract.org/2004/07/”不是预料之中的。   考虑使用DataContractResolver或添加任何未知的类型   静态地到已知类型的列表 - 例如,通过使用   KnownTypeAttribute属性或通过将它们添加到已知列表中   传递给DataContractSerializer的类型。

我已经尝试使用GlobalConfiguration.Configuration.Formatters.XmlFormatter.SetSerializer为我知道的设置断点的生成类型设置序列化程序,但它似乎忽略它并抛出相同的异常。枚举将由整数支持,并保证每个条目都有唯一值。这是我用来生成类型并返回它的实例的代码。

public object GetSerializableEnumProxy( Type enumType ) {

    if ( enumType == null ) {
        throw new ArgumentNullException( "enumType" );
    }

    if ( !enumType.IsEnum ) {
        throw new InvalidOperationException();
    }

    AssemblyName assemblyName = new AssemblyName("DataBuilderAssembly");
    AssemblyBuilder assemBuilder = AppDomain.CurrentDomain.DefineDynamicAssembly(assemblyName, AssemblyBuilderAccess.Run);
    ModuleBuilder moduleBuilder = assemBuilder.DefineDynamicModule("DataBuilderModule");
    TypeBuilder typeBuilder = moduleBuilder.DefineType( enumType.Name, TypeAttributes.Class | TypeAttributes.Public );

    // Add the [DataContract] attribute to our generated type
    typeBuilder.SetCustomAttribute(
        new CustomAttributeBuilder( typeof(DataContractAttribute).GetConstructor( Type.EmptyTypes ), new object[] {} )
    );

    CustomAttributeBuilder dataMemberAttributeBuilder = new CustomAttributeBuilder(
        typeof(DataMemberAttribute).GetConstructor(Type.EmptyTypes), new object[] {}
    );

    // For each name in the enum, define a corresponding public int field
    // with the [DataMember] attribute
    foreach ( var value in Enum.GetValues(enumType).Cast<int>() ) {
        var name = Enum.GetName( enumType, value );

        var fb = typeBuilder.DefineField( name, typeof(int), FieldAttributes.Public );

        // Add the [DataMember] attribute to the field
        fb.SetCustomAttribute( dataMemberAttributeBuilder );

        // Set the value of our field to be the corresponding value from the Enum
        fb.SetConstant( value );
    }       

    // Return an instance of our generated type
    return Activator.CreateInstance( typeBuilder.CreateType() );
}

Web Api控制器方法:

private static IEnumerable<Type> RetrievableEnums = new Type[] {
    typeof(Priority), typeof(Status)
};

[GET("enum/{enumName}")]
public HttpResponseMessage GetEnumInformation( string enumName ) {

    Type enumType = RetrievableEnums.SingleOrDefault( type =>
        String.Equals( type.Name, enumName, StringComparison.InvariantCultureIgnoreCase));

    if ( enumType == null ) {
        return Request.CreateErrorResponse( HttpStatusCode.NotFound, "The requested enum could not be retrieved" );
    }

    return Request.CreateResponse( HttpStatusCode.OK, GetSerializableEnumProxy(enumType) );
}

有什么想法吗?

1 个答案:

答案 0 :(得分:8)

我相信这最终是因为你将枚举值发送为object - 与Json格式化程序不同,Web API的xml格式化程序使用DataContractSerializer,使用(实际上)编译 - 序列化值的时间类型,而不是运行时类型。

因此,您必须始终确保将您尝试序列化的任何派生类型的基础添加到基础序列化程序的已知类型中。在这种情况下,您有动态枚举(当然是object)。

从表面上看,似乎SetSerializer(type, serializer)方法应该有效,但是,我敢打赌你用动态类型作为第一个参数来调用它 - 如果你那将不起作用正在将枚举发送为object - 因为它是object将使用的XmlRequestFormatter序列化程序。

这是一个众所周知的问题 - 一个which I've reported as an issue on codeplex(这里有一个很好的例子,可以在更简单的场景中演示问题)。

该问题还包括一些属性的C#代码和XmlMediaTypeFormatter(称为XmlMediaTypeFormatterEx)的替换,它为此问题提供了一种解决方案 - 它使用声明式按操作方法。将XmlMediaTypeFormatter替换为代码中的一个 - 使用类似的东西(注意这段代码处理的情况是没有定义XML格式化程序 - 可能有些毫无意义):

var configuration = GlobalConfiguration.Configuration;  
var origXmlFormatter = configuration.Formatters.OfType<XmlMediaTypeFormatter>()
                       .SingleOrDefault();

XmlMediaTypeFormatterEx exXmlFormatter = new XmlMediaTypeFormatterEx(origXmlFormatter);

if (origXmlFormatter != null)
{
    configuration.Formatters.Insert(
      configuration.Formatters.IndexOf(origXmlFormatter), exXmlFormatter);
    configuration.Formatters.Remove(origXmlFormatter);
}
else
    configuration.Formatters.Add(exXmlFormatter);

现在,在您想要返回此动态枚举的API方法上,您可以用它来装饰它:

[XmlUseReturnedUnstanceType]
public object Get()
{

}

现在,无论您从Get方法返回什么类型,自定义格式化程序都会启动并专门针对运行时类型使用DataContractSerializer,而不是object

这不处理基数的可枚举或字典 - 它变得非常复杂 - 但对于基本的单实例返回值,它可以正常工作。