Pet*_*ete 2 c# asp.net-web-api
我有一个带有几个控制器的WebApi,它们会返回不同的结果.例如,一个控制器返回一个IEnumberable,另一个控制器返回一个Bar,另一个控制器返回IEnumberable等IEnumberable等,我所要做的就是:
return Ok(thething)
Run Code Online (Sandbox Code Playgroud)
一切正常,甚至复杂的嵌套对象都被序列化没有问题.
现在,客户端要求在Wrapper中返回所有结果:
public class Wrapper
{
public bool Success { get; set; }
public int ErrorCode { get; set; }
public String ErrorMessage { get; set; }
public String Referer { get; set; }
public Object Payload { get; set; }
}
Run Code Online (Sandbox Code Playgroud)
认为这将是微不足道的,但当我尝试从控制器返回时:
return Ok( new Wrapper { Success=true, Referer="me", Payload=thething)
Run Code Online (Sandbox Code Playgroud)
我收到序列化错误.
异常消息是:
'ObjectContent`1'类型无法序列化内容类型'application/xml的响应主体; 字符集= UTF-8' .
内部异常消息是:
输入'System.Linq.Enumerable + WhereSelectListIterator
2[[EPiServer.Find.Api.SearchHit1 [[DGTNext.Api.Data.Entities.ProductSummary,DGTNext.Api.Entities,Version = 1.0.0.0,Culture = neutral,PublicKeyToken = null]],EPiServer.Find,Version = 9.6.0.3185,Culture = neutral,PublicKeyToken = 8fe83dea738b45b7],[DGTNext.Api.Data.Entities.ProductSummary,DGTNext.Api.Entities,Version = 1.0.0.0,Culture = neutral,PublicKeyToken = null]]',带有数据合约名称'ArrayOfProductSummary:http://schemas.datacontract.org/2004/07/DGTNext.Api.Data.Entities '不是预期的.如果您正在使用DataContractSerializer或者将任何静态未知的类型添加到已知类型列表中,请考虑使用DataContractResolver - 例如,通过使用KnownTypeAttribute属性或将它们添加到传递给序列化程序的已知类型列表中.
我究竟做错了什么?为什么Ok()函数之前似乎处理任何对象,但现在有问题?
谢谢.
编辑:根据要求,一个导致错误的简单示例:
class Foo {
public int AnInt { get; set; }
}
public IHttpActionResult Get() {
return Ok(new Wrapper { Success=true, Referer="me", Payload= new Foo {AnInt = 7}});
}
Run Code Online (Sandbox Code Playgroud)
编辑:
好吧,我提出了一种解决方案,但它仍然提出了一些问题.
我在有效载荷的类型中使我的Wrapper通用.
public class Wrapper<T>
{
public bool Success { get; set; }
public int ErrorCode { get; set; }
public String ErrorMessage { get; set; }
public String Referer { get; set; }
public T Payload { get; set; }
}
Run Code Online (Sandbox Code Playgroud)
所以现在,这有效:
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, Referer="me", Payload= foos});
}
Run Code Online (Sandbox Code Playgroud)
它返回:
{"Success":true,"ErrorCode":0,"ErrorMessage":null,"Referer":"me","Payload":[{"AnInt":7},{"AnInt":8}]}
Run Code Online (Sandbox Code Playgroud)
而我的"真实"电话:
public IHttpActionResult Get() {
IEnumerable<ProductSummary> prods = db.getProductSummaries(shopId, culture, queryParams, paging);
return Ok(new Wrapper<IEnumerable<ProductSummary>> { Success = true, Referer = "me", Payload = prods });
}
Run Code Online (Sandbox Code Playgroud)
收益:
<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.
Run Code Online (Sandbox Code Playgroud)
所以不错,但这提出了两个问题:
为了测试webapi,我通过在Firefox地址栏中放置一个URL并在浏览器中查看结果来调用它.为什么世界上第一次调用返回Json,第二次调用XML?据我所知,我只是使用默认的一切.
为什么XML现在将该命名空间添加到所有元素名称?我可以阻止这个吗?当我在没有包装器的情况下返回相同的东西时,这并没有发生.
将此代码添加到Application_Start下面的global.asax:
从.Ignore更新为.Serialize.它必须工作.
GlobalConfiguration.Configuration.Formatters.JsonFormatter.SerializerSettings.ReferenceLoopHandling = Newtonsoft.Json.ReferenceLoopHandling.Serialize;
GlobalConfiguration.Configuration.Formatters.Remove(GlobalConfiguration.Configuration.Formatters.XmlFormatter);
Run Code Online (Sandbox Code Playgroud)
或者你可以看看这个答案
| 归档时间: |
|
| 查看次数: |
1467 次 |
| 最近记录: |