Nasıl yapılır: Bir gecikmeden sonra tamamlanan bir görev oluşturma
Bu örnekte, bir gecikmeden concurrency::task
sonra tamamlanan bir görev oluşturmak için , concurrency::cancellation_token_source
concurrency::cancellation_token
, concurrency::task_completion_event
, concurrency::timer
ve concurrency::call
sınıflarının nasıl kullanılacağı gösterilmektedir. Verileri ara sıra yoklayan döngüler oluşturmak için bu yöntemi kullanabilirsiniz. Ayrıca zaman aşımları, önceden belirlenmiş bir süre için kullanıcı girişinin gecikmeli işlenmesi gibi işlemleri de tanıtabilirsiniz.
Örnek: complete_after ve cancel_after_timeout işlevleri
Aşağıdaki örnekte ve cancel_after_timeout
işlevleri gösterilmektedircomplete_after
. işlevi, complete_after
belirtilen gecikmeden sonra tamamlanan bir task
nesne oluşturur. Belirtilen gecikmeden sonra bir timer
nesne ayarlamak için bir nesnesi ve call
nesnesi task_completion_event
kullanır. sınıfını task_completion_event
kullanarak, bir iş parçacığından veya başka bir görev bir değerin kullanılabilir olduğunu işaret ettikten sonra tamamlayan bir görev tanımlayabilirsiniz. Olay ayarlandığında dinleyici görevleri tamamlanır ve devamları çalıştırılacak şekilde zamanlanır.
İpucu
Zaman Uyumsuz Aracılar Kitaplığı'nın timer
parçası olan ve call
sınıfları hakkında daha fazla bilgi için bkz . Zaman Uyumsuz İleti Blokları.
İşlev, cancel_after_timeout
belirli bir zaman aşımından complete_after
önce bu görev tamamlanmazsa, görevi iptal etmek için işlevi üzerinde derler. cancel_after_timeout
işlevi iki görev oluşturur. İlk görev başarıyı gösterir ve sağlanan görev tamamlandıktan sonra tamamlar. İkinci görev hata gösterir ve belirtilen zaman aşımından sonra tamamlar. İşlev, cancel_after_timeout
başarı veya başarısızlık görevi tamamlandığında çalışan bir devamlılık görevi oluşturur. İlk olarak hata görevi tamamlanırsa, devamlılık belirteç kaynağını iptal eder ve genel görevi iptal eder.
// Creates a task that completes after the specified delay.
task<void> complete_after(unsigned int timeout)
{
// A task completion event that is set when a timer fires.
task_completion_event<void> tce;
// Create a non-repeating timer.
auto fire_once = new timer<int>(timeout, 0, nullptr, false);
// Create a call object that sets the completion event after the timer fires.
auto callback = new call<int>([tce](int)
{
tce.set();
});
// Connect the timer to the callback and start the timer.
fire_once->link_target(callback);
fire_once->start();
// Create a task that completes after the completion event is set.
task<void> event_set(tce);
// Create a continuation task that cleans up resources and
// and return that continuation task.
return event_set.then([callback, fire_once]()
{
delete callback;
delete fire_once;
});
}
// Cancels the provided task after the specifed delay, if the task
// did not complete.
template<typename T>
task<T> cancel_after_timeout(task<T> t, cancellation_token_source cts, unsigned int timeout)
{
// Create a task that returns true after the specified task completes.
task<bool> success_task = t.then([](T)
{
return true;
});
// Create a task that returns false after the specified timeout.
task<bool> failure_task = complete_after(timeout).then([]
{
return false;
});
// Create a continuation task that cancels the overall task
// if the timeout task finishes first.
return (failure_task || success_task).then([t, cts](bool success)
{
if(!success)
{
// Set the cancellation token. The task that is passed as the
// t parameter should respond to the cancellation and stop
// as soon as it can.
cts.cancel();
}
// Return the original task.
return t;
});
}
Örnek: Asal sayıların işlem sayısı
Aşağıdaki örnek [0, 100000] aralığındaki asal sayıların sayısını birden çok kez hesaplar. Belirli bir süre sınırında tamamlanmazsa işlem başarısız olur. işlevi, count_primes
işlevinin cancel_after_timeout
nasıl kullanılacağını gösterir. Verilen aralıktaki asal değerleri sayar ve görev belirtilen sürede tamamlanmazsa başarısız olur. İşlev işlevi wmain
birden çok kez çağırır count_primes
. Her seferinde zaman sınırını yarıya indirer. İşlem geçerli zaman sınırında tamamlanamadıktan sonra program tamamlanmıştır.
// Determines whether the input value is prime.
bool is_prime(int n)
{
if (n < 2)
return false;
for (int i = 2; i < n; ++i)
{
if ((n % i) == 0)
return false;
}
return true;
}
// Counts the number of primes in the range [0, max_value].
// The operation fails if it exceeds the specified timeout.
bool count_primes(unsigned int max_value, unsigned int timeout)
{
cancellation_token_source cts;
// Create a task that computes the count of prime numbers.
// The task is canceled after the specified timeout.
auto t = cancel_after_timeout(task<size_t>([max_value, timeout, cts]
{
combinable<size_t> counts;
parallel_for<unsigned int>(0, max_value + 1, [&counts, cts](unsigned int n)
{
// Respond if the overall task is cancelled by canceling
// the current task.
if (cts.get_token().is_canceled())
{
cancel_current_task();
}
// NOTE: You can replace the calls to is_canceled
// and cancel_current_task with a call to interruption_point.
// interruption_point();
// Increment the local counter if the value is prime.
if (is_prime(n))
{
counts.local()++;
}
});
// Return the sum of counts across all threads.
return counts.combine(plus<size_t>());
}, cts.get_token()), cts, timeout);
// Print the result.
try
{
auto primes = t.get();
wcout << L"Found " << primes << L" prime numbers within "
<< timeout << L" ms." << endl;
return true;
}
catch (const task_canceled&)
{
wcout << L"The task timed out." << endl;
return false;
}
}
int wmain()
{
// Compute the count of prime numbers in the range [0, 100000]
// until the operation fails.
// Each time the test succeeds, the time limit is halved.
unsigned int max = 100000;
unsigned int timeout = 5000;
bool success = true;
do
{
success = count_primes(max, timeout);
timeout /= 2;
} while (success);
}
/* Sample output:
Found 9592 prime numbers within 5000 ms.
Found 9592 prime numbers within 2500 ms.
Found 9592 prime numbers within 1250 ms.
Found 9592 prime numbers within 625 ms.
The task timed out.
*/
Bir gecikmeden sonra görevleri iptal etmek için bu tekniği kullandığınızda, genel görev iptal edildikten sonra başlatılmamış görevler başlatılmaz. Ancak, uzun süre çalışan görevlerin iptale hızlı bir şekilde yanıt vermesi önemlidir. Görev iptali hakkında daha fazla bilgi için bkz . PPL'de İptal.
Tam kod örneği
Bu örneğin tam kodu aşağıda verilmiştir:
// task-delay.cpp
// compile with: /EHsc
#include <ppl.h>
#include <ppltasks.h>
#include <agents.h>
#include <iostream>
using namespace concurrency;
using namespace std;
// Creates a task that completes after the specified delay.
task<void> complete_after(unsigned int timeout)
{
// A task completion event that is set when a timer fires.
task_completion_event<void> tce;
// Create a non-repeating timer.
auto fire_once = new timer<int>(timeout, 0, nullptr, false);
// Create a call object that sets the completion event after the timer fires.
auto callback = new call<int>([tce](int)
{
tce.set();
});
// Connect the timer to the callback and start the timer.
fire_once->link_target(callback);
fire_once->start();
// Create a task that completes after the completion event is set.
task<void> event_set(tce);
// Create a continuation task that cleans up resources and
// and return that continuation task.
return event_set.then([callback, fire_once]()
{
delete callback;
delete fire_once;
});
}
// Cancels the provided task after the specifed delay, if the task
// did not complete.
template<typename T>
task<T> cancel_after_timeout(task<T> t, cancellation_token_source cts, unsigned int timeout)
{
// Create a task that returns true after the specified task completes.
task<bool> success_task = t.then([](T)
{
return true;
});
// Create a task that returns false after the specified timeout.
task<bool> failure_task = complete_after(timeout).then([]
{
return false;
});
// Create a continuation task that cancels the overall task
// if the timeout task finishes first.
return (failure_task || success_task).then([t, cts](bool success)
{
if(!success)
{
// Set the cancellation token. The task that is passed as the
// t parameter should respond to the cancellation and stop
// as soon as it can.
cts.cancel();
}
// Return the original task.
return t;
});
}
// Determines whether the input value is prime.
bool is_prime(int n)
{
if (n < 2)
return false;
for (int i = 2; i < n; ++i)
{
if ((n % i) == 0)
return false;
}
return true;
}
// Counts the number of primes in the range [0, max_value].
// The operation fails if it exceeds the specified timeout.
bool count_primes(unsigned int max_value, unsigned int timeout)
{
cancellation_token_source cts;
// Create a task that computes the count of prime numbers.
// The task is canceled after the specified timeout.
auto t = cancel_after_timeout(task<size_t>([max_value, timeout, cts]
{
combinable<size_t> counts;
parallel_for<unsigned int>(0, max_value + 1, [&counts, cts](unsigned int n)
{
// Respond if the overall task is cancelled by canceling
// the current task.
if (cts.get_token().is_canceled())
{
cancel_current_task();
}
// NOTE: You can replace the calls to is_canceled
// and cancel_current_task with a call to interruption_point.
// interruption_point();
// Increment the local counter if the value is prime.
if (is_prime(n))
{
counts.local()++;
}
});
// Return the sum of counts across all threads.
return counts.combine(plus<size_t>());
}, cts.get_token()), cts, timeout);
// Print the result.
try
{
auto primes = t.get();
wcout << L"Found " << primes << L" prime numbers within "
<< timeout << L" ms." << endl;
return true;
}
catch (const task_canceled&)
{
wcout << L"The task timed out." << endl;
return false;
}
}
int wmain()
{
// Compute the count of prime numbers in the range [0, 100000]
// until the operation fails.
// Each time the test succeeds, the time limit is halved.
unsigned int max = 100000;
unsigned int timeout = 5000;
bool success = true;
do
{
success = count_primes(max, timeout);
timeout /= 2;
} while (success);
}
/* Sample output:
Found 9592 prime numbers within 5000 ms.
Found 9592 prime numbers within 2500 ms.
Found 9592 prime numbers within 1250 ms.
Found 9592 prime numbers within 625 ms.
The task timed out.
*/
Kod Derleniyor
Kodu derlemek için kopyalayın ve visual studio projesine yapıştırın veya adlı task-delay.cpp
bir dosyaya yapıştırın ve ardından visual studio komut istemi penceresinde aşağıdaki komutu çalıştırın.
cl.exe /EHsc task-delay.cpp
Ayrıca bkz.
Görev Paralelliği
task Sınıfı (Eşzamanlılık Çalışma Zamanı)
cancellation_token_source Sınıfı
cancellation_token Sınıfı
task_completion_event Sınıfı
timer Sınıfı
call Sınıfı
Zaman Uyumsuz İleti Blokları
PPL'de İptal