ASP.NET SCRIPTERVICE PROBLEMA DE DESERIALIZAÇÃO COM TIPOS DERIDOS
-
26-09-2019 - |
Pergunta
eu tenho um ScriptService
Método da Web (.NET 3.5) que leva um único parâmetro de um tipo de base abstrato:
[WebMethod(EnableSession=true)]
[ScriptMethod()]
public bool Test(Item item) { ... }
E:
namespace Namespace {
public abstract class Item
{
public int id;
}
public class Group : Item
{
public Item[] items;
}
public class Instance : Item
{
public string whatever;
}
}
Geralmente, quando o método é chamado, item
será uma Group
que contém Instance
e/ou Group
objetos. Estou chamando esse serviço da JQuery; Não estou usando a estrutura do lado do cliente da Microsoft. As chamadas para outros métodos funcionam bem.
O problema: Quando faço a ligação, uma exceção é lançada antes mesmo de meu método ser invocado. Por exemplo, se minha chamada for:
POST /WebService.asmx/Test HTTP/1.1
Content-Type: application/json; charset=UTF-8
Accept: application/json, text/javascript, */*
{"item":{"id":0,"__type":"Namespace.Group","items":[]}}
... eu entendo um InvalidOperationException
:
{"Message":"Operation is not valid due to the current state of the object.","StackTrace":" at System.Web.Script.Serialization.ObjectConverter.ConvertDictionaryToObject(IDictionary`2 dictionary, Type type, JavaScriptSerializer serializer, Boolean throwOnError, Object& convertedObject)\r\n at System.Web.Script.Serialization.ObjectConverter.ConvertObjectToTypeInternal(Object o, Type type, JavaScriptSerializer serializer, Boolean throwOnError, Object& convertedObject)\r\n at System.Web.Script.Serialization.ObjectConverter.ConvertObjectToTypeMain(Object o, Type type, JavaScriptSerializer serializer, Boolean throwOnError, Object& convertedObject)\r\n at System.Web.Script.Serialization.JavaScriptObjectDeserializer.DeserializeInternal(Int32 depth)\r\n at System.Web.Script.Serialization.JavaScriptObjectDeserializer.DeserializeDictionary(Int32 depth)\r\n at System.Web.Script.Serialization.JavaScriptObjectDeserializer.DeserializeInternal(Int32 depth)\r\n at System.Web.Script.Serialization.JavaScriptObjectDeserializer.BasicDeserialize(String input, Int32 depthLimit, JavaScriptSerializer serializer)\r\n at System.Web.Script.Serialization.JavaScriptSerializer.Deserialize(JavaScriptSerializer serializer, String input, Type type, Int32 depthLimit)\r\n at System.Web.Script.Serialization.JavaScriptSerializer.Deserialize[T](String input)\r\n at System.Web.Script.Services.RestHandler.ExecuteWebServiceCall(HttpContext context, WebServiceMethodData methodData)","ExceptionType":"System.InvalidOperationException"}
Se eu soltar o __type
membro do objeto JSON ou altere -o para Namespace.Item
(e remova o abstract
modificador de Item
), a exceção desaparece, mas o objeto desserializado resultante é obviamente meio inútil.
o que estou perdendo?
Solução
Ah ha! Clicando em torno dos links relacionados no lado direito desta pergunta, eu encontrei uma resposta Isso me ajudou a resolver esse problema.
o GenerateScriptType
O atributo deve ser aplicado à classe de serviço da web:
[WebService( ... )]
[WebServiceBinding(ConformsTo = WsiProfiles.BasicProfile1_1)]
[ScriptService]
[GenerateScriptType(typeof(Group))]
[GenerateScriptType(typeof(Instance))]
public class WebService : System.Web.Services.WebService
{
[WebMethod(EnableSession=true)]
[ScriptMethod()]
public bool Test(Item item) { ... }
}
Sem esses atributos, o Deserializer não conhece meus tipos derivados.
Outras dicas
Eu poderia resolver isso usando Newtonsofts JsonConvert
como abaixo.
[System.Web.Services.WebMethod]
public static void MyAjaxCallingMethod(object inputs)
{
string stingDataFromAjaxCall = JsonConvert.SerializeObject(inputs);
MyObject myObj = JsonConvert.DeserializeObject<MyObject>(stingDataFromAjaxCall);
....}