Как исправить ошибку MaxItemsInObjectGraph?
У меня есть служба RESTful WCF, и я получаю следующую ошибку: Максимальное количество элементов, которые можно сериализовать или десериализовать в графе объектов, равно "65536". Измените граф объекта или увеличьте квоту MaxItemsInObjectGraph.
Я думал, что решил это, но, видимо, нет. Вот мой код:
Я использую файл.SVC, который использует пользовательскую фабрику, как это:
<%@ ServiceHost Language="C#" Debug="true" Service="myService" Factory="myCustomWebServiceHostFactory" %>
Вот код для кастомной фабрики
public class myCustomWebServiceHost : WebServiceHost
{
public myCustomWebServiceHost()
{
}
public myCustomWebServiceHost(object singletonInstance, params Uri[] baseAddresses)
: base(singletonInstance, baseAddresses)
{
}
public myCustomWebServiceHost(Type serviceType, params Uri[] baseAddresses)
: base(serviceType, baseAddresses)
{
}
protected override void OnOpening()
{
foreach (var endpoint in Description.Endpoints)
{
var binding = endpoint.Binding as WebHttpBinding;
if (binding != null)
{
const int fiveMegaBytes = 5242880;
binding.MaxReceivedMessageSize = fiveMegaBytes;
binding.MaxBufferSize = fiveMegaBytes;
binding.MaxBufferPoolSize = fiveMegaBytes;
}
}
base.OnOpening();
}
}
class myCustomWebServiceHostFactory : WebServiceHostFactory
{
protected override ServiceHost CreateServiceHost(Type serviceType, Uri[] baseAddresses)
{
return new myCustomWebServiceHost(serviceType, baseAddresses);
}
}
Обслуживание:
[AspNetCompatibilityRequirements(RequirementsMode = AspNetCompatibilityRequirementsMode.Allowed)]
[ServiceContract]
[ServiceBehavior(MaxItemsInObjectGraph = int.MaxValue)]
public class myService
{
...service implementation code goes here
}
Клиент:
public class myClient
{
WebChannelFactory<IMyService> cf;
IMyService channel;
public myClient()
{
WebHttpBinding _binding = new WebHttpBinding();
_binding.MaxBufferPoolSize = 5000000;
_binding.MaxBufferSize = 5000000;
_binding.MaxReceivedMessageSize = 5000000;
_binding.TransferMode = TransferMode.Streamed;
_binding.ReceiveTimeout = new TimeSpan(0, 0, 30);
_binding.ReaderQuotas.MaxArrayLength = 5000000;
Uri _uri = new Uri("http://myserviceurl");
cf = new WebChannelFactory<IMyService>(_binding, _uri);
channel = cf.CreateChannel();
foreach (OperationDescription op in cf.Endpoint.Contract.Operations)
{
var dataContractBehavior = op.Behaviors.Find<DataContractSerializerOperationBehavior>();
if (dataContractBehavior != null)
{
dataContractBehavior.MaxItemsInObjectGraph = int.MaxValue;
}
}
}
...client implementation code goes here
}
ОБНОВЛЕНИЕ: я сделал небольшое устранение неполадок. Кажется, что часть сериализации работает нормально на сервере. Я могу выполнить GET из браузера и получить обратно все данные в формате XML. Так что это работает нормально. Кажется, это часть десериализации, которая вызывает ошибку. Если вы посмотрите выше в коде для myClient, вы увидите, как я пытаюсь установить свойство MaxItemsInObjectGraph для поведения DataContractSerializer. Я делаю это правильно?
4 ответа
Я понял!!! Мой код клиента был неверным. Я устанавливал MaxItemsInObjectGraph после того, как я уже создал свой канал. Таким образом, свойство MaxItemInObjectGraph не влияло на уже созданный канал. (Этот материал WCF меня так смущает - я обычно просто копирую и вставляю код, не зная, что на самом деле делаю) Вот исправленный код:
WebHttpBinding _binding = new WebHttpBinding();
_binding.MaxBufferPoolSize = 5000000;
_binding.MaxBufferSize = 5000000;
_binding.MaxReceivedMessageSize = 5000000;
_binding.TransferMode = TransferMode.Streamed;
_binding.ReceiveTimeout = new TimeSpan(0, 0, 30);
_binding.ReaderQuotas.MaxArrayLength = 5000000;
Uri _uri = new Uri(http://myserviceurl);
cf = new WebChannelFactory<IMyService>(_binding, _uri);
foreach (OperationDescription op in cf.Endpoint.Contract.Operations)
{
var dataContractBehavior = op.Behaviors.Find<DataContractSerializerOperationBehavior>();
if (dataContractBehavior != null)
{
dataContractBehavior.MaxItemsInObjectGraph = int.MaxValue;
}
}
channel = cf.CreateChannel();
Вам необходимо установить MaxItemsInObjectGraph для dataContractSerializer, используя поведение как клиента, так и службы. Смотрите здесь для примера.
В ваших запросах вы возвращаете общий список или массив, размер которого превышает 65536. Использование топ-60000 или отсутствие добавления более 60 тыс. Элементов решит вашу проблему.
На стороне клиента еще один способ достичь этого - создать собственную реализацию IEndPointBehavior - см. Http://canbilgin.wordpress.com/2010/06/25/how-to-set-maxitemsinobjectgraph-programmatics-for-client/,
Это особенно полезно при реализации клиента с Windsor WcfFacility
public class ReaderQuotaExtension : IEndpointBehavior
{
#region Implementation of IEndpointBehavior
public void Validate(ServiceEndpoint endpoint)
{
}
public void AddBindingParameters(ServiceEndpoint endpoint, BindingParameterCollection bindingParameters)
{
}
public void ApplyDispatchBehavior(ServiceEndpoint endpoint, EndpointDispatcher endpointDispatcher)
{
ModifyDataContractSerializerBehavior(endpoint);
}
public void ApplyClientBehavior(ServiceEndpoint endpoint, ClientRuntime clientRuntime)
{
ModifyDataContractSerializerBehavior(endpoint);
}
#endregion
private static void ModifyDataContractSerializerBehavior(ServiceEndpoint endpoint)
{
foreach (var behavior in endpoint.Contract.Operations.Select(operation => operation.Behaviors.Find<DataContractSerializerOperationBehavior>()))
{
behavior.MaxItemsInObjectGraph = 2147483647;
}
}