Как исправить ошибку 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;
        }
    }
Другие вопросы по тегам