我有一个带有多个返回不同结果的控制器的 WebApi。例如,一个控制器返回了IEnumerable<Foo>
,另一个Bar
,另一个IEnumerable
of IEnumerable
等等,而我所要做的就是:
return Ok(thething)
一切工作正常,即使是复杂的嵌套对象也可以毫无问题地序列化。
现在,客户要求所有结果都在包装器中返回:
public class Wrapper
{
public bool Success { get; set; }
public int ErrorCode { get; set; }
public String ErrorMessage { get; set; }
public String Referrer { get; set; }
public Object Payload { get; set; }
}
认为这很简单,但是当我尝试从控制器返回它时:
return Ok(new Wrapper { Success = true, Referrer = "me", Payload = thething)
我收到序列化错误:
“ObjectContent`1”类型无法序列化内容类型“application/xml”的响应正文;字符集=utf-8'。
内部异常消息是:
'System.Linq.Enumerable+WhereSelectListIterator2[[EPiServer.Find.Api.SearchHit
1[[DGTNext.Api.Data.Entities.ProductSummary,
DGTNext.Api.Entities,版本=1.0.0.0,文化=中性,
PublicKeyToken=null]],EPiServer.Find,版本=9.6.0.3185,
文化=中立,
PublicKeyToken=8fe83dea738b45b7],[DGTNext.Api.Data.Entities.ProductSummary,
DGTNext.Api.Entities,版本=1.0.0.0,文化=中性,
PublicKeyToken=null]]' 带有数据合约名称
'ArrayOfProductSummary:http://schemas.datacontract.org/2004/07/DGTNext.Api.Data.Entities' http://schemas.datacontract.org/2004/07/DGTNext.Api.Data.Entities%27预计不会。如果您是,请考虑使用 DataContractResolver
使用 DataContractSerializer 或添加任何静态未知的类型
已知类型的列表 - 例如,通过使用 KnownTypeAttribute
属性或将它们添加到传递给的已知类型列表中
序列化器。
我究竟做错了什么?为什么Ok()
函数以前似乎可以处理任何对象,但现在有问题吗?
Thanks.
编辑:根据要求,导致错误的简单示例:
class Foo
{
public int AnInt { get; set; }
}
public IHttpActionResult Get()
{
return Ok(new Wrapper { Success = true, Referrer = "me", Payload = new Foo { AnInt = 7 } });
}
Edit #2:嗯,我想出了一种解决方案,但它仍然提出了一些问题。
我使我的包装器在有效负载类型中通用。
public class Wrapper<T>
{
public bool Success { get; set; }
public int ErrorCode { get; set; }
public String ErrorMessage { get; set; }
public String Referrer { get; set; }
public T Payload { get; set; }
}
所以现在,这有效:
public IHttpActionResult Get()
{
List<Foo> foos = new List<Foo>();
foos.Add(new Foo { AnInt = 7 });
foos.Add(new Foo { AnInt = 8 });
return Ok(new Wrapper<IEnumerable<Foo>> { Success = true, Referrer = "me", Payload = foos });
}
它返回:
{
"Success": true,
"ErrorCode": 0,
"ErrorMessage": null,
"Referrer": "me",
"Payload": [ { "AnInt": 7 }, { "AnInt": 8 } ]
}
我的“真实”电话:
public IHttpActionResult Get()
{
IEnumerable<ProductSummary> prods = db.getProductSummaries(shopId, culture, queryParams, paging);
return Ok(new Wrapper<IEnumerable<ProductSummary>> { Success = true, Referrer = "me", Payload = prods });
}
returns:
<WrapperOfArrayOfProductSummaryzc2y5_Pnl>
<ErrorCode>0</ErrorCode>
<ErrorMessage i:nil="true"/>
<Payload>
<d2p1:ProductSummary>
<d2p1:Culture i:nil="true"/>
<d2p1:Guid i:nil="true"/>
<d2p1:Id>2</d2p1:Id>
<d2p1:Name>Letto Asia</d2p1:Name>
<d2p1:ambient>
<d2p1:Id>1073741838</d2p1:Id>
<d2p1:Name>notte</d2p1:Name>
</d2p1:ambient>
etc.
所以还不错,但这提出了两个问题:
-
为了测试 webapi,我通过在 Firefox 地址栏中输入 URL 并在浏览器中查看结果来调用它。为什么第一个调用返回 Json,第二个调用返回 XML?据我所知,我只是使用默认的一切。
-
为什么 XML 现在要将该命名空间添加到所有元素名称中?我可以阻止这种情况吗?当我只是在没有包装的情况下返回相同的东西时,这种情况并没有发生。