Używanie metod asynchronicznych na platformie ASP.NET 4.5
Autor : Rick Anderson
Ten samouczek zawiera podstawowe informacje na temat tworzenia aplikacji ASP.NET Web Forms asynchronicznej przy użyciu Visual Studio Express 2012 for Web, która jest bezpłatną wersją programu Microsoft Visual Studio. Możesz również użyć programu Visual Studio 2012. Poniższe sekcje znajdują się w tym samouczku.
- Jak żądania są przetwarzane przez pulę wątków
- Wybieranie metod synchronicznych lub asynchronicznych
- Przykładowa aplikacja
- Strona synchroniczna Gizmos
- Tworzenie asynchronicznej strony gizmos
- Równoległe wykonywanie wielu operacji
- Używanie tokenu anulowania
- Konfiguracja serwera dla wywołań usługi sieci Web o dużym opóźnieniu/dużych współbieżności
Kompletny przykład jest udostępniany na potrzeby tego samouczka pod adresem
https://github.com/RickAndMSFT/Async-ASP.NET/ w witrynie GitHub .
ASP.NET 4.5 Web Pages w połączeniu z platformą .NET 4.5 umożliwia rejestrowanie metod asynchronicznych, które zwracają obiekt typu Task. W .NET Framework 4 wprowadzono asynchroniczną koncepcję programowania nazywaną zadaniem, a ASP.NET 4.5 obsługuje zadanie. Zadania są reprezentowane przez typ zadania i powiązane typy w przestrzeni nazw System.Threading.Tasks . .NET Framework 4.5 opiera się na tej asynchronicznej obsłudze słów kluczowych await i asynchronicznych, które sprawiają, że praca z obiektami zadań jest znacznie mniej złożona niż poprzednie podejścia asynchroniczne. Słowo kluczowe await to skrót składniowy wskazujący, że fragment kodu powinien asynchronicznie czekać na inny fragment kodu. Słowo kluczowe asynchroniczne reprezentuje wskazówkę, której można użyć do oznaczania metod jako metod asynchronicznych opartych na zadaniach. Kombinacja funkcji await, asynchronicznego i obiektu Task znacznie ułatwia pisanie kodu asynchronicznego na platformie .NET 4.5. Nowy model metod asynchronicznych jest nazywany asynchronicznym wzorcem asynchronicznym opartym na zadaniach (TAP). W tym samouczku założono, że masz pewną znajomość programowania asynchronicznego przy użyciu słów kluczowych await i asynchronicznych oraz przestrzeni nazw Zadania .
Aby uzyskać więcej informacji na temat używania słów kluczowych await i asynchronicznych oraz przestrzeni nazw zadania , zobacz następujące odwołania.
- Oficjalny dokument: Asynchronia na platformie .NET
- Async/Await — często zadawane pytania
- Programowanie asynchroniczne programu Visual Studio
Jak żądania są przetwarzane przez pulę wątków
Na serwerze internetowym .NET Framework obsługuje pulę wątków używanych do obsługi żądań ASP.NET. Po odebraniu żądania jest wysyłany wątek z puli w celu przetworzenia tego żądania. Jeśli żądanie jest przetwarzane synchronicznie, wątek, który przetwarza żądanie, jest zajęty podczas przetwarzania żądania, a ten wątek nie może obsłużyć innego żądania.
Może to nie być problem, ponieważ pula wątków może być wystarczająco duża, aby pomieścić wiele zajętych wątków. Jednak liczba wątków w puli wątków jest ograniczona (wartość domyślna maksymalna dla platformy .NET 4.5 to 5000). W dużych aplikacjach z wysoką współbieżnością długotrwałych żądań wszystkie dostępne wątki mogą być zajęte. Ten warunek jest nazywany głodem wątku. Po osiągnięciu tego warunku serwer internetowy kolejkuje żądania. Jeśli kolejka żądań stanie się pełna, serwer internetowy odrzuca żądania ze stanem HTTP 503 (Serwer jest zbyt zajęty). Pula wątków CLR ma ograniczenia dotyczące iniekcji nowego wątku. Jeśli współbieżność jest pęknięta (oznacza to, że witryna internetowa może nagle uzyskać dużą liczbę żądań), a wszystkie dostępne wątki żądań są zajęte z powodu wywołań zaplecza z dużym opóźnieniem, ograniczona szybkość iniekcji wątku może sprawić, że aplikacja będzie reagować bardzo źle. Ponadto każdy nowy wątek dodany do puli wątków ma narzut (na przykład 1 MB pamięci stosu). Aplikacja internetowa korzystająca z metod synchronicznych do obsługi wywołań o dużym opóźnieniu, w których pula wątków zwiększa się do domyślnego maksymalnie 5,000 wątków platformy .NET 4.5, 000 wątków zużywałoby około 5 GB więcej pamięci niż aplikacja mogła obsługiwać te same żądania przy użyciu metod asynchronicznych i tylko 50 wątków. Podczas wykonywania pracy asynchronicznej nie zawsze używasz wątku. Na przykład po utworzeniu asynchronicznego żądania usługi internetowej ASP.NET nie będzie używać żadnych wątków między wywołaniem metody asynchronicznej a await. Użycie puli wątków do żądań obsługi z dużym opóźnieniem może prowadzić do dużego zużycia pamięci i słabego wykorzystania sprzętu serwera.
Przetwarzanie żądań asynchronicznych
W aplikacjach internetowych, które widzą dużą liczbę współbieżnych żądań podczas uruchamiania lub mają zwiększone obciążenie (w przypadku nagłego wzrostu współbieżności), asynchroniczne wywołania usług internetowych zwiększają czas odpowiedzi aplikacji. Żądanie asynchroniczne trwa tyle samo czasu, aby przetwarzać je jako żądanie synchroniczne. Jeśli na przykład żądanie wykonuje wywołanie usługi internetowej, które wymaga wykonania dwóch sekund, żądanie trwa dwie sekundy, niezależnie od tego, czy jest wykonywane synchronicznie, czy asynchronicznie. Jednak podczas wywołania asynchronicznego wątek nie może odpowiadać na inne żądania, czekając na ukończenie pierwszego żądania. W związku z tym żądania asynchroniczne uniemożliwiają kolejkowanie żądań i wzrost puli wątków, gdy istnieje wiele współbieżnych żądań, które wywołują długotrwałe operacje.
Wybieranie metod synchronicznych lub asynchronicznych
W tej sekcji wymieniono wskazówki dotyczące używania metod synchronicznych lub asynchronicznych. Są to tylko wytyczne; zbadaj każdą aplikację indywidualnie, aby określić, czy metody asynchroniczne pomagają w wydajności.
Ogólnie rzecz biorąc, użyj metod synchronicznych dla następujących warunków:
- Operacje są proste lub krótkie.
- Prostota jest ważniejsza niż wydajność.
- Operacje to przede wszystkim operacje procesora CPU, a nie operacje obejmujące duże obciążenie dysku lub sieci. Korzystanie z metod asynchronicznych w operacjach związanych z procesorem CPU nie zapewnia żadnych korzyści i powoduje większe obciążenie.
Ogólnie rzecz biorąc, użyj metod asynchronicznych dla następujących warunków:
Wywołujesz usługi, które mogą być używane za pomocą metod asynchronicznych i używasz platformy .NET 4.5 lub nowszej.
Operacje są powiązane z siecią lub operacjami we/wy zamiast powiązanymi z procesorem CPU.
Równoległość jest ważniejsza niż prostota kodu.
Chcesz udostępnić mechanizm, który umożliwia użytkownikom anulowanie długotrwałego żądania.
Gdy korzyść z przełączania wątków przewyższa koszt przełącznika kontekstu. Ogólnie rzecz biorąc, należy wykonać metodę asynchroniczną, jeśli metoda synchroniczna blokuje wątek żądania ASP.NET, nie wykonując żadnej pracy. Wykonując wywołanie asynchroniczne, wątek żądania ASP.NET nie jest blokowany, nie wykonuje żadnej pracy podczas oczekiwania na ukończenie żądania usługi internetowej.
Testowanie pokazuje, że operacje blokowania są wąskim gardłem w wydajności lokacji i że usługi IIS mogą obsługiwać więcej żądań przy użyciu metod asynchronicznych dla tych wywołań blokujących.
W przykładzie do pobrania pokazano, jak efektywnie używać metod asynchronicznych. Udostępniony przykład został zaprojektowany w celu zapewnienia prostego pokazu programowania asynchronicznego w ASP.NET 4.5. Przykład nie jest przeznaczony do architektury referencyjnej programowania asynchronicznego w ASP.NET. Przykładowy program wywołuje metody internetowego interfejsu API ASP.NET , które z kolei wywołają metodę Task.Delay w celu symulowania długotrwałych wywołań usług internetowych. Większość aplikacji produkcyjnych nie pokaże takich oczywistych korzyści z używania metod asynchronicznych.
Niewiele aplikacji wymaga, aby wszystkie metody są asynchroniczne. Często konwertowanie kilku metod synchronicznych na metody asynchroniczne zapewnia najlepszy wzrost wydajności dla wymaganej ilości pracy.
Przykładowa aplikacja
Przykładową aplikację można pobrać z https://github.com/RickAndMSFT/Async-ASP.NET witryny GitHub . Repozytorium składa się z trzech projektów:
- WebAppAsync: projekt ASP.NET Web Forms korzystający z usługi WebAPIpwg internetowego interfejsu API. Większość kodu dla tego samouczka pochodzi z tego projektu.
- WebAPIpgw: projekt internetowego interfejsu
Products, Gizmos and Widgets
API MVC 4 ASP.NET implementujący kontrolery. Udostępnia ona dane dla projektu WebAppAsync i projektu Mvc4Async . - Mvc4Async: projekt ASP.NET MVC 4 zawierający kod używany w innym samouczku. Wykonuje ona wywołania internetowego interfejsu API do usługi WebAPIpwg .
Strona synchroniczna Gizmos
Poniższy kod przedstawia metodę Page_Load
synchroniczną używaną do wyświetlania listy gizmos. (W tym artykule gizmo jest fikcyjnym urządzeniem mechanicznym).
public partial class Gizmos : System.Web.UI.Page
{
protected void Page_Load(object sender, EventArgs e)
{
var gizmoService = new GizmoService();
GizmoGridView.DataSource = gizmoService.GetGizmos();
GizmoGridView.DataBind();
}
}
Poniższy kod przedstawia metodę GetGizmos
usługi gizmo.
public class GizmoService
{
public async Task<List<Gizmo>> GetGizmosAsync(
// Implementation removed.
public List<Gizmo> GetGizmos()
{
var uri = Util.getServiceUri("Gizmos");
using (WebClient webClient = new WebClient())
{
return JsonConvert.DeserializeObject<List<Gizmo>>(
webClient.DownloadString(uri)
);
}
}
}
Metoda GizmoService GetGizmos
przekazuje identyfikator URI do usługi HTTP internetowego interfejsu API ASP.NET, która zwraca listę danych gizmos. Projekt WebAPIpgw zawiera implementację internetowego interfejsu API gizmos, widget
i product
kontrolerów.
Na poniższej ilustracji przedstawiono stronę gizmos z przykładowego projektu.
Tworzenie asynchronicznej strony gizmos
W przykładzie użyto nowych słów kluczowych asynchronicznych i await (dostępnych w programach .NET 4.5 i Visual Studio 2012), aby umożliwić kompilatorowi zachowanie skomplikowanych przekształceń niezbędnych do programowania asynchronicznego. Kompilator umożliwia pisanie kodu przy użyciu synchronicznych konstrukcji przepływu sterowania języka C#, a kompilator automatycznie stosuje przekształcenia niezbędne do używania wywołań zwrotnych w celu uniknięcia blokowania wątków.
ASP.NET strony asynchroniczne muszą zawierać dyrektywę Page z atrybutem ustawionym Async
na wartość "true". Poniższy kod przedstawia dyrektywę Page z atrybutem ustawionym Async
na wartość "true" dla strony GizmosAsync.aspx .
<%@ Page Async="true" Language="C#" AutoEventWireup="true"
CodeBehind="GizmosAsync.aspx.cs" Inherits="WebAppAsync.GizmosAsync" %>
Poniższy kod przedstawia metodę Gizmos
synchroniczną Page_Load
i GizmosAsync
stronę asynchroniczną. Jeśli przeglądarka obsługuje element znacznika> HTML 5<, zobaczysz zmiany w GizmosAsync
żółtym wyróżnieniu.
protected void Page_Load(object sender, EventArgs e)
{
var gizmoService = new GizmoService();
GizmoGridView.DataSource = gizmoService.GetGizmos();
GizmoGridView.DataBind();
}
Wersja asynchroniczna:
protected void Page_Load(object sender, EventArgs e)
{
RegisterAsyncTask(new PageAsyncTask(GetGizmosSvcAsync));
}
private async Task GetGizmosSvcAsync()
{
var gizmoService = new GizmoService();
GizmosGridView.DataSource = await gizmoService.GetGizmosAsync();
GizmosGridView.DataBind();
}
Zastosowano następujące zmiany, aby zezwolić na GizmosAsync
asynchroniczną stronę.
- Dyrektywa Page musi mieć
Async
atrybut ustawiony na "true". - Metoda
RegisterAsyncTask
służy do rejestrowania asynchronicznego zadania zawierającego kod, który jest uruchamiany asynchronicznie. - Nowa
GetGizmosSvcAsync
metoda jest oznaczona za pomocą słowa kluczowego asynchronicznego , które nakazuje kompilatorowi generowanie wywołań zwrotnych dla części treści i automatyczne utworzenieTask
zwracanego elementu. - Element "Async" został dołączony do nazwy metody asynchronicznej. Dołączanie metody asynchronicznej nie jest wymagane, ale jest konwencją podczas pisania metod asynchronicznych.
- Zwracany typ nowej
GetGizmosSvcAsync
metody toTask
. ZwracanyTask
typ reprezentuje bieżącą pracę i udostępnia obiekt wywołujący metodę z uchwytem, za pomocą którego należy poczekać na zakończenie operacji asynchronicznej. - Słowo kluczowe await zostało zastosowane do wywołania usługi internetowej.
- Wywołano asynchroniczny interfejs API usługi internetowej (
GetGizmosAsync
).
GetGizmosSvcAsync
Wewnątrz metody jest wywoływana inna metoda GetGizmosAsync
asynchroniczna. GetGizmosAsync
natychmiast zwraca element Task<List<Gizmo>>
, który zostanie ostatecznie ukończony po udostępnieniu danych. Ponieważ nie chcesz robić nic innego, dopóki nie masz danych gizmo, kod czeka na zadanie (przy użyciu słowa kluczowego await ). Słowo kluczowe await można używać tylko w metodach z adnotacjami ze słowem kluczowym asynchronicznego .
Słowo kluczowe await nie blokuje wątku do momentu ukończenia zadania. Spowoduje to zarejestrowanie pozostałej części metody jako wywołania zwrotnego w zadaniu i natychmiastowe zwrócenie. Po zakończeniu oczekiwanego zadania wywołanie zwrotne wywołania zwrotnego spowoduje wznowienie wykonywania metody w prawo od lewej. Aby uzyskać więcej informacji na temat używania słów kluczowych await i asynchronicznych oraz przestrzeni nazw zadań , zobacz odwołania asynchroniczne.
W poniższym kodzie przedstawiono metody GetGizmos
i GetGizmosAsync
.
public List<Gizmo> GetGizmos()
{
var uri = Util.getServiceUri("Gizmos");
using (WebClient webClient = new WebClient())
{
return JsonConvert.DeserializeObject<List<Gizmo>>(
webClient.DownloadString(uri)
);
}
}
public async Task<List<Gizmo>> GetGizmosAsync()
{
var uri = Util.getServiceUri("Gizmos");
using (WebClient webClient = new WebClient())
{
return JsonConvert.DeserializeObject<List<Gizmo>>(
await webClient.DownloadStringTaskAsync(uri)
);
}
}
Zmiany asynchroniczne są podobne do zmian wprowadzonych w powyższej konfiguracji GizmosAsync .
- Sygnatura metody została oznaczona adnotacją za pomocą słowa kluczowego asynchronicznego , typ zwracany został zmieniony na
Task<List<Gizmo>>
, a Async został dołączony do nazwy metody. - Asynchroniczna klasa HttpClient jest używana zamiast synchronicznej klasy WebClient .
- Słowo kluczowe await zostało zastosowane do metody asynchronicznej HttpClientGetAsync .
Na poniższej ilustracji przedstawiono asynchroniczny widok gizmo.
Prezentacja przeglądarki danych gizmos jest identyczna z widokiem utworzonym przez wywołanie synchroniczne. Jedyna różnica polega na tym, że wersja asynchroniczna może być bardziej wydajna w przypadku ciężkich obciążeń.
RegisterAsyncTask Notes
Metody podłączone za pomocą RegisterAsyncTask
będą uruchamiane natychmiast po preRender.
Jeśli bezpośrednio używasz zdarzeń strony asynchronicznego pustki, jak pokazano w poniższym kodzie:
protected async void Page_Load(object sender, EventArgs e) {
await ...;
// do work
}
Nie masz już pełnej kontroli nad wykonywaniem zdarzeń. Jeśli na przykład zarówno plik aspx, jak i . Definiowanie Page_Load
zdarzeń wzorca i jedna lub obie z nich są asynchroniczne, kolejność wykonywania nie może być gwarantowana. Ta sama nieokreślona kolejność dla programów obsługi zdarzeń (takich jak async void Button_Click
) ma zastosowanie.
Wykonywanie wielu operacji równolegle
Metody asynchroniczne mają znaczącą przewagę nad metodami synchronicznymi, gdy akcja musi wykonywać kilka niezależnych operacji. W podanym przykładzie synchroniczna strona PWG.aspx(dla produktów, widżetów i Gizmos) wyświetla wyniki trzech wywołań usługi internetowej, aby uzyskać listę produktów, widżetów i gizmos. Projekt internetowego interfejsu API ASP.NET , który udostępnia te usługi, używa funkcji Task.Delay do symulowania opóźnienia lub wolnych wywołań sieciowych. Gdy opóźnienie jest ustawione na 500 milisekund, asynchroniczna strona PWGasync.aspx zajmuje nieco ponad 500 milisekund do ukończenia, podczas gdy wersja synchroniczna PWG
przejmuje ponad 1500 milisekund. Synchroniczna strona PWG.aspx jest wyświetlana w poniższym kodzie.
protected void Page_Load(object sender, EventArgs e)
{
Stopwatch stopWatch = new Stopwatch();
stopWatch.Start();
var widgetService = new WidgetService();
var prodService = new ProductService();
var gizmoService = new GizmoService();
var pwgVM = new ProdGizWidgetVM(
widgetService.GetWidgets(),
prodService.GetProducts(),
gizmoService.GetGizmos()
);
WidgetGridView.DataSource = pwgVM.widgetList;
WidgetGridView.DataBind();
ProductGridView.DataSource = pwgVM.prodList;
ProductGridView.DataBind();
GizmoGridView.DataSource = pwgVM.gizmoList;
GizmoGridView.DataBind();
stopWatch.Stop();
ElapsedTimeLabel.Text = String.Format("Elapsed time: {0}",
stopWatch.Elapsed.Milliseconds / 1000.0);
}
Poniżej przedstawiono kod asynchroniczny PWGasync
.
protected void Page_Load(object sender, EventArgs e)
{
Stopwatch stopWatch = new Stopwatch();
stopWatch.Start();
RegisterAsyncTask(new PageAsyncTask(GetPWGsrvAsync));
stopWatch.Stop();
ElapsedTimeLabel.Text = String.Format("Elapsed time: {0}",
stopWatch.Elapsed.Milliseconds / 1000.0);
}
private async Task GetPWGsrvAsync()
{
var widgetService = new WidgetService();
var prodService = new ProductService();
var gizmoService = new GizmoService();
var widgetTask = widgetService.GetWidgetsAsync();
var prodTask = prodService.GetProductsAsync();
var gizmoTask = gizmoService.GetGizmosAsync();
await Task.WhenAll(widgetTask, prodTask, gizmoTask);
var pwgVM = new ProdGizWidgetVM(
widgetTask.Result,
prodTask.Result,
gizmoTask.Result
);
WidgetGridView.DataSource = pwgVM.widgetList;
WidgetGridView.DataBind();
ProductGridView.DataSource = pwgVM.prodList;
ProductGridView.DataBind();
GizmoGridView.DataSource = pwgVM.gizmoList;
GizmoGridView.DataBind();
}
Na poniższej ilustracji przedstawiono widok zwrócony ze strony asynchronicznej PWGasync.aspx .
Korzystanie z tokenu anulowania
Metody asynchroniczne zwracane Task
są do anulowania, czyli przyjmują parametr CancelToken , gdy jest dostarczany z AsyncTimeout
atrybutem dyrektywy Page . Poniższy kod przedstawia stronę GizmosCancelAsync.aspx z limitem czasu na sekundę.
<%@ Page Async="true" AsyncTimeout="1"
Language="C#" AutoEventWireup="true"
CodeBehind="GizmosCancelAsync.aspx.cs"
Inherits="WebAppAsync.GizmosCancelAsync" %>
Poniższy kod przedstawia plik GizmosCancelAsync.aspx.cs .
protected void Page_Load(object sender, EventArgs e)
{
RegisterAsyncTask(new PageAsyncTask(GetGizmosSvcCancelAsync));
}
private async Task GetGizmosSvcCancelAsync(CancellationToken cancellationToken)
{
var gizmoService = new GizmoService();
var gizmoList = await gizmoService.GetGizmosAsync(cancellationToken);
GizmosGridView.DataSource = gizmoList;
GizmosGridView.DataBind();
}
private void Page_Error(object sender, EventArgs e)
{
Exception exc = Server.GetLastError();
if (exc is TimeoutException)
{
// Pass the error on to the Timeout Error page
Server.Transfer("TimeoutErrorPage.aspx", true);
}
}
W podanej przykładowej aplikacji wybranie linku GizmosCancelAsync wywołuje stronę GizmosCancelAsync.aspx i demonstruje anulowanie (według limitu czasu) wywołania asynchronicznego. Ponieważ czas opóźnienia mieści się w zakresie losowym, może być konieczne odświeżenie strony kilka razy, aby uzyskać komunikat o błędzie przekroczenia limitu czasu.
Konfiguracja serwera dla wywołań usługi sieci Web o wysokim opóźnieniu/dużym opóźnieniu
Aby zrealizować korzyści wynikające z asynchronicznej aplikacji internetowej, może być konieczne wprowadzenie pewnych zmian w domyślnej konfiguracji serwera. Pamiętaj o następujących kwestiach podczas konfigurowania i testowania obciążenia aplikacji internetowej asynchronicznej.
Systemy operacyjne Windows 7, Windows Vista, Windows 8 i wszystkie systemy operacyjne klienckie systemu Windows mają maksymalnie 10 równoczesnych żądań. Potrzebujesz systemu operacyjnego Windows Server, aby zobaczyć zalety metod asynchronicznych pod dużym obciążeniem.
Zarejestruj program .NET 4.5 z usługami IIS w wierszu polecenia z podwyższonym poziomem uprawnień, używając następującego polecenia:
%windir%\Microsoft.NET\Framework64 \v4.0.30319\aspnet_regiis -i
Zobacz ASP.NET narzędzie rejestracji usług IIS (Aspnet_regiis.exe)Może być konieczne zwiększenie limitu kolejkiHTTP.sys z wartości domyślnej 1000 do 5000. Jeśli ustawienie jest zbyt niskie, może zostać wyświetlone HTTP.sys odrzucanie żądań ze stanem HTTP 503. Aby zmienić limit kolejki HTTP.sys:
- Otwórz menedżera usług IIS i przejdź do okienka Pule aplikacji.
- Kliknij prawym przyciskiem myszy docelową pulę aplikacji i wybierz pozycję Ustawienia zaawansowane.
- W oknie dialogowym Ustawienia zaawansowane zmień długość kolejki z 1000 na 5000.
Uwaga na powyższych obrazach platforma .NET Framework jest wyświetlana jako wersja 4.0, mimo że pula aplikacji korzysta z platformy .NET 4.5. Aby zrozumieć tę rozbieżność, zobacz następujące kwestie:
How to set an IIS Application or AppPool to use ASP.NET 3.5 a nie 2.0
Jeśli aplikacja korzysta z usług internetowych lub System.NET do komunikowania się z zapleczem za pośrednictwem protokołu HTTP, może być konieczne zwiększenie elementu connectionManagement/maxconnection . W przypadku aplikacji ASP.NET jest to ograniczone przez funkcję autokonfiguracji do 12 razy więcej procesorów CPU. Oznacza to, że w quad-proc można mieć co najwyżej 12 * 4 = 48 współbieżnych połączeń z punktem końcowym adresu IP. Ponieważ jest to powiązane z autoConfig, najprostszym sposobem zwiększenia
maxconnection
ASP.NET aplikacji jest ustawienie parametru System.Net.ServicePointManager.DefaultConnectionLimit programowo w metodzie fromApplication_Start
w pliku global.asax . Zobacz przykład pobierania przykładu.W programie .NET 4.5 wartość domyślna 5000 dla parametru MaxConcurrentRequestsPerCPU powinna być odpowiednia.