Передача большого объема данных в службе WCF
Я создал веб-службу в WCF, которая возвращает более 54000 строк данных с 10 данными в каждой строке. Я использовал привязка wshttpbinding для общения. Служба хорошо работает с меньшим количеством данных (т. е. 2000 строк), но она бомбит при попытке отправить большой набор записей с 50000+ строк (~2 МБ). Сообщение об исключении выглядит следующим образом
произошла ошибка при получении HTTP-ответа на
http://localhost:9002/MyService.svc
. Это может быть связано с тем, что привязка конечной точки службы не использование протокола HTTP. Это также может быть связано с тем, что контекст HTTP-запроса прерывается сервером (возможно, из-за отключения службы). Дополнительные сведения см. В разделе журналы сервера.
пожалуйста, не говорите мне использовать разбиение на страницы на стороне клиента - я знаю, что это решит проблему. Но мне нужен весь кусок данных в клиентской части.
настройки сервиса на сервер как
<system.serviceModel>
<bindings>
<wsHttpBinding>
<binding name="MyWsHttpBinding" />
</wsHttpBinding>
</bindings>
<services>
<service name="AdminService">
<endpoint address="AdminSrv"
binding="wsHttpBinding"
contract="IAdminService"/>
<endpoint address="mex" binding="mexHttpBinding" contract="IMetadataExchange" />
<host>
<baseAddresses>
<add baseAddress="/Bus/IRfotoWCF" />
</baseAddresses>
</host>
</service>
</services>
<behaviors>
<serviceBehaviors>
<behavior>
<!-- To avoid disclosing metadata information,
set the value below to false and remove the metadata endpoint above before deployment -->
<serviceMetadata httpGetEnabled="True"/>
<!-- To receive exception details in faults for debugging purposes,
set the value below to true. Set to false before deployment
to avoid disclosing exception information -->
<serviceDebug includeExceptionDetailInFaults="True" />
</behavior>
</serviceBehaviors>
</behaviors>
<serviceHostingEnvironment multipleSiteBindingsEnabled="true"></serviceHostingEnvironment>
</system.serviceModel>
моя конфигурация клиента как
<system.serviceModel>
<bindings>
<basicHttpBinding>
<binding name="BasicHttpBinding_IAdminService" closeTimeout="00:01:00"
openTimeout="00:01:00" receiveTimeout="00:10:00" sendTimeout="00:01:00"
allowCookies="false" bypassProxyOnLocal="false" hostNameComparisonMode="StrongWildcard"
maxBufferSize="65536" maxBufferPoolSize="524288" maxReceivedMessageSize="65536"
messageEncoding="Text" textEncoding="utf-8" transferMode="Buffered"
useDefaultWebProxy="true">
<readerQuotas maxDepth="32" maxStringContentLength="8192" maxArrayLength="16384"
maxBytesPerRead="4096" maxNameTableCharCount="16384" />
<security mode="None">
<transport clientCredentialType="None" proxyCredentialType="None" realm="" />
<message clientCredentialType="UserName" algorithmSuite="Default" />
</security>
</binding>
</basicHttpBinding>
</bindings>
<client>
<endpoint address="http://localhost/TestService/AdminService.svc"
binding="basicHttpBinding" bindingConfiguration="BasicHttpBinding_IAdminService"
contract="IAdminService" name="BasicHttpBinding_IAdminService" />
</client>
</system.serviceModel>
кто-нибудь поможет мне с конфигурацией excact как на стороне клиента, так и на стороне сервера. Даже если мне нужно изменить привязку с привязка wshttpbinding to привязки nettcpbinding - у меня нет проблем с этим. Спасибо заранее.
3 ответов
после большого расследования finnally я получил решение. На самом деле многое нужно изменить.
следующие изменения необходимо сделать в на стороне сервера.
первый я должен был назначить maxRequestLength для большего значения в my строки httpruntime элемент для запуска запроса на более длительный период.
<system.web>
<httpRuntime maxRequestLength="102400" />
</system.web>
второй я представил привязки nettcpbinding binnding с пользовательскими изменениями на maxBufferSize, maxBufferPoolSize, maxReceivedMessageSize
С большим значением 2147483647
.
<binding name="myNetTcpBinding"
maxBufferPoolSize="2147483647"
maxBufferSize="524288"
maxReceivedMessageSize="2147483647">
третий добавить maxItemsInObjectGraph
в обоих serviceBehaviors
и endpointBehaviors
как ниже (не забудьте упомянуть имена поведения в service
и endpoint
узел)
<behaviors>
<serviceBehaviors>
<behavior name="myNetTcpBehaviour">
<serviceMetadata httpGetEnabled="true"/>
<serviceDebug includeExceptionDetailInFaults="true"/>
<dataContractSerializer maxItemsInObjectGraph="2147483647"/>
</behavior>
</serviceBehaviors>
<endpointBehaviors>
<behavior name="myNetTcpEndPointBehaviour">
<dataContractSerializer maxItemsInObjectGraph="2147483647"/>
</behavior>
</endpointBehaviors>
</behaviors>
наконец-то моя конфигурация сервера выглядит так
<system.web>
<httpRuntime maxRequestLength="102400" />
</system.web>
<system.serviceModel>
<bindings>
<wsHttpBinding>
<binding name="MyWsHttpBinding" />
</wsHttpBinding>
<netTcpBinding>
<binding name="myNetTcpBinding"
closeTimeout="00:01:00"
openTimeout="00:01:00"
receiveTimeout="00:10:00"
sendTimeout="00:01:00"
transactionFlow="false"
transferMode="Buffered"
transactionProtocol="OleTransactions"
hostNameComparisonMode="StrongWildcard"
listenBacklog="10"
maxBufferPoolSize="2147483647"
maxBufferSize="524288"
maxConnections="10"
maxReceivedMessageSize="2147483647">
<readerQuotas maxDepth="32"
maxStringContentLength="8192"
maxArrayLength="16384"
maxBytesPerRead="4096"
maxNameTableCharCount="16384" />
<reliableSession ordered="true"
inactivityTimeout="00:10:00"
enabled="false" />
<security mode="Transport">
<transport clientCredentialType="Windows" protectionLevel="EncryptAndSign" />
</security>
</binding>
</netTcpBinding>
</bindings>
<services>
<service name="AdminService" behaviorConfiguration="myNetTcpBehaviour">
<endpoint address="AdminSrv"
binding="netTcpBinding"
bindingConfiguration="myNetTcpBinding"
contract="IAdminService"
behaviorConfiguration="myNetTcpEndPointBehaviour"/>
<endpoint address="mex" binding="mexHttpBinding" contract="IMetadataExchange" />
<host>
<baseAddresses>
<add baseAddress="/Bus/IRfotoWCF" />
</baseAddresses>
</host>
</service>
<behaviors>
<serviceBehaviors>
<behavior name="myNetTcpBehaviour">
<serviceMetadata httpGetEnabled="true"/>
<serviceDebug includeExceptionDetailInFaults="true"/>
<dataContractSerializer maxItemsInObjectGraph="2147483647"/>
</behavior>
</serviceBehaviors>
<endpointBehaviors>
<behavior name="myNetTcpEndPointBehaviour">
<dataContractSerializer maxItemsInObjectGraph="2147483647"/>
</behavior>
</endpointBehaviors>
</behaviors>
<serviceHostingEnvironment multipleSiteBindingsEnabled="true"></serviceHostingEnvironment>
</system.serviceModel>
теперь о на стороне клиента configuratioin вам нужно изменить maxBufferSize="2147483647" maxBufferPoolSize="524288" maxReceivedMessageSize="2147483647"
и также вам нужно добавить maxItemsInObjectGraph="2147483647"
в конфигурации поведения конечной точки.
<endpointBehaviors>
<behavior name="myEndPointBehavior">
<dataContractSerializer maxItemsInObjectGraph="2147483647" />
</behavior>
</endpointBehaviors>
теперь я могу передать 30000 строка 5.30 мин где запрос выполняется в течение 10 сек, поэтому время передачи составляет 5.20 мин -еще много.
не стесняйтесь комментировать и любые предложения по улучшению.
Если вы посмотрите на детали привязки, они не совпадают полностью на сервере и на стороне клиента. Атрибуты для maxBufferSize, maxBufferPoolSize, maxReceivedMessageSize
также должны быть определены на стороне сервера. А затем вам нужно поместить значения в соответствии с размером, на который вы смотрите.
вместо использования for loop over WCF для объемных данных используйте пользовательский тип таблицы (если вы используете SQL) . Это позволит сократить время с 6 мин до 15-20 сек.