传递具有继承对象值的基础对象(Json)



我调用了一个WebAPI,代码如下:

var client = new HttpClient
{
    BaseAddress = new Uri("http://localhost:8490/")
};
var jObject = new JObject();
jObject.Add("paramA", paramA);
jObject.Add("paramB", paramB);
JArray jArr = JArray.FromObject(paramsGenericArr);
jObject.Add("paramC", jArr);
var content = new StringContent(jObject.ToString(), Encoding.UTF8, "application/json");
var result = await client.PostAsync("api/path/tofunc", content).ConfigureAwait(false);
result.EnsureSuccessStatusCode();

ParamsGeneric类是一个抽象类型,有两个派生类:

[DataContract]
public class ParamsTypeA : ParamsGeneric
{
    [DataMember]
    public long itemC {get; set;}
    public ParamsTypeA() :
                      base()
    {}
}
[DataContract]
public class ParamsTypeB : ParamsGeneric
{
    [DataMember]
    public long itemD {get; set;}
    public ParamsTypeB() :
                      base()
    {}
}
[DataContract]
[KnownType(typeof(ParamsTypeA))]
[KnownType(typeof(ParamsTypeB))]
public abstract class ParamsGeneric
{
    [DataMember]
    public long itemA { get; set; }
    [DataMember]
    public long itemB {get; set;}
    public ParamsGeneric() 
    {}
}

我怀疑我在WebAPI:中的反序列化有问题

public class ClientData
{
    public string paramA { get; set; }
    public string paramB { get; set; }
    public ParamsGeneric[] paramC { get; set; }
}
[HttpPost]
[Route("api/path/tofunc")]
public async Task<bool> DoStuffAsync(ClientData clientData)
{
    ....
}

我对paramsGenericArr/paramC(类型为ParamsGeneric[],包含类型为ParamsTypeA&ParamsTypeB的项目)有问题

WebAPI接收一个空白数组(ParamsGeneric[0])以及其他参数。

将通知帮助。

更新

即使我尝试传递单个ParamsGeneric对象而不是数组,我也会接收null而不是对象。

解决方案

var serializer = new JsonSerializer();
serializer.TypeNameHandling = TypeNameHandling.Auto;
JArray jArr = JArray.FromObject(paramsGenericArr, serializer);

成功了。

虽然messages/json中的继承是肯定可能的,但IMHO;这太麻烦了:)

不管怎样,你实际上可以让Newtonsoft。Json通过设置TypeNameHandling 为您处理继承

// Allow inheritance in json payload
JsonSerializerSettings serializerSettings = GlobalConfiguration.Configuration.Formatters.JsonFormatter.SerializerSettings;
serializerSettings.TypeNameHandling = TypeNameHandling.All;

或者只是

serializerSettings.TypeNameHandling = TypeNameHandling.Auto;

根据您的需要。

这是一个简单的解决方案,如果它是一个内部API,或者如果你可以保证你将始终控制客户端,这将很好地工作。如果您有外部客户端,我会使用"覆盖默认模型绑定器"(override default model binder)方法,比如这里发布的"将Json取消为Asp.Net Web API中的派生类型",或者非常强烈地考虑避免在API的模型中继承。

尝试将数据传递到您的API,如下

Dictionary<string, string> param = new Dictionary<string, string>();
param.Add("paramA", paramA);
param.Add("paramB", paramB);
HttpClient client = new HttpClient();
HttpFormUrlEncodedContent contents = new HttpFormUrlEncodedContent(param);
var result = await client.PostAsync(new Uri("http://localhost:8490/api/path/tofunc")
                                                                         , contents);
var reply = await result.Content.ReadAsStringAsync();
if (reply.IsSuccessStatusCode)
{ 
}

希望这对你有帮助。

[HttpPost]
[Route("api/path/tofunc")]
public async Task<bool> DoStuffAsync([FromBody]ClientData clientData)
{
....
}

请在web api方法中保留[FromBody],以便模型绑定器将您的身体数据映射到参数,即clientData。

相关内容

  • 没有找到相关文章

最新更新