我调用了一个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。