Запретить многопоточному веб-сайту слишком много ресурсов - PullRequest
3 голосов
/ 09 марта 2012

Я создал веб-сайт для массовой рассылки электронной почты для клиента, который должен отправить 80 000 писем за одну отправку.По сути, он создает новый поток для отправки, так что управление может быть передано обратно в пользовательский интерфейс (чтобы можно было загрузить страницу обратной связи), а затем для каждой компании создается новый поток, чтобы отправлять электронные письма их получателям.Все электронные письма помещаются в очередь с использованием этого кода:

// Loop through the companies and send their mail to the specified recipients
        // while creating a new thread for each company
        // A new thread is started so that the feedback page can load
        SendingThread = Task.Factory.StartNew(() =>
        {
            // This is a thread safe for loop
            Parallel.ForEach<CompanyEntity>(companies, company =>
            {
                    // Start a new thread for each company send
                    Task.Factory.StartNew(() =>
                    {
                        // Get the recipients for this company
                        var companyRecipients = GetSubscribersForCompany(company.Id, recipients);

                        // Send the newsletter to the company recipients
                        var success = SendNewsletterForCompany(newsletter, company, companyRecipients, language,
                                                               version, company.AdvertCollectionViaNewsletterCompanyAdvertLink, newsletter.NewsletterType, email);

                        // Add the status update so the front end can view a list of updated conpany statuses
                        if (success)
                            AddStatusUpdate(company.CompanyTitle + " has completed processing.");

                        // Starts sending the emails if the engine hasn't already been started
                        SendEngine.Start(CurrentSmtpClient, this);

                    }).ContinueWith(antecendent => EndCompaniesSendUpdate(companiesToProcess, companiesProcessed), TaskContinuationOptions.OnlyOnRanToCompletion);
            });
        }, new CancellationToken(), TaskCreationOptions.LongRunning, TaskScheduler.Default);

Когда электронные письма помещаются в очередь, механизм отправки берет на себя и извлекает электронные письма из очереди, а затем отправляет их, используя новый класс Parallel:

Action action = () =>
        {
            MailMessage message;
            while (queue.TryDequeue(out message))
            {
                SendMessage(sendingServer, message, factory);
            }
        };

        // Start 5 concurrent actions to send the messages in parallel.
        Parallel.Invoke(action, action, action, action, action);

Все это прекрасно работает и может отослать 40 000 рассылок за 10 минут.Единственная проблема заключается в том, что ОЗУ и ЦП на сервере потребляются на 100% в течение этих 10 минут.Это влияет на другие сайты на сервере, так как к ним нельзя получить доступ.

Есть ли способ ограничить использование ресурсов отправляющим приложением в IIS 7.5 или путем изменения кода выше?

1 Ответ

2 голосов
/ 14 марта 2012

Проблемы:

  • Вы создаете поток внутри Parallel ForEach. Часть «Параллель» означает, что она уже порождает поток для тела. Вы вкладываете Parallel Invoke вместо действия внутри Parallel ForEach внутри другого действия.

  • Вы выполняете цикл while внутри потока без отдыха ЦП. Параллельно вызывается 5x.

Ответы:

Для использования процессора вам нужно дать вашей обработке передышку. В вашем цикле While TryDequeue поместите короткий сон.

        MailMessage message;
        while (queue.TryDequeue(out message))
        {
            SendMessage(sendingServer, message, factory);
            Thread.Sleep(16);
        }

Для использования ОЗУ и ЦП необходимо обрабатывать LESS одновременно.

        SendingThread = Task.Factory.StartNew(() =>
        {
            foreach(var company in companies) 
            {
                        // Get the recipients for this company
                        var companyRecipients = GetSubscribersForCompany(company.Id, recipients);

                        // Send the newsletter to the company recipients
                        var success = SendNewsletterForCompany(newsletter, company, companyRecipients, language,
                                                               version, company.AdvertCollectionViaNewsletterCompanyAdvertLink, newsletter.NewsletterType, email);

                        // Add the status update so the front end can view a list of updated conpany statuses
                        if (success)
                            AddStatusUpdate(company.CompanyTitle + " has completed processing.");

                        // Starts sending the emails if the engine hasn't already been started
                        SendEngine.Start(CurrentSmtpClient, this);


            }
       }, new CancellationToken(), TaskCreationOptions.LongRunning, TaskScheduler.Default);
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...