Asynchronous Tasking
This chapters discusses how to launch tasks asynchronously so that you can incorporate independent, dynamic parallelism in your taskflows.
Launch Asynchronous Tasks from an Executor
Taskflow executor provides an STL-styled method, tf::
std::future<int> future = executor.async([](){ return 1; }); assert(future.get() == 1);
If you do not need the return value or use a future to synchronize the execution, you are encouraged to use tf::
executor.silent_async([](){ // do some work without returning any result });
Launching asynchronous tasks from an executor is thread-safe and can be called by multiple threads both inside (i.e., worker) and outside the executor. Our scheduler autonomously detects whether an asynchronous task is submitted from an external thread or a worker thread and schedules its execution using work stealing.
tf::Task my_task = taskflow.emplace([&](){ // launch an asynchronous task from my_task executor.async([&](){ // launch another asynchronous task that may be run by another worker executor.async([&](){}); }) }); executor.run(taskflow); executor.wait_for_all(); // wait for all tasks to finish
You can name an asynchronous task using the overloads, tf::Executor::async(const std::string& name, F&& f) and tf::Executor::silent_async(const std::string& name, F&& f), that take a string in the first argument. Assigned names will appear in the observers of the executor.
std::future<void> fu = executor.async("async task", [](){}); executor.silent_async("silent async task", [](){});
Launch Asynchronous Tasks from a Subflow
You can launch asynchronous tasks from tf::
tf::Taskflow taskflow; tf::Executor executor; std::atomic<int> counter{0}; taskflow.emplace([&] (tf::Subflow& sf){ std::vector<std::future<void>> futures; for(int i=0; i<100; i++) { futures.emplace_back(sf.async([&](){ ++counter; })); } sf.join(); // all of the 100 asynchronous tasks will finish by this join assert(counter == 100); }); executor.run(taskflow).wait();
If you do not need the return value or the future to synchronize the execution, you can use tf::
tf::Taskflow taskflow; tf::Executor executor; std::atomic<int> counter{0}; taskflow.emplace([&] (tf::Subflow& sf){ for(int i=0; i<100; i++) { sf.silent_async([&](){ ++counter; }); } sf.join(); // all of the 100 asynchronous tasks will finish by this join assert(counter == 100); }); executor.run(taskflow).wait();
You can assign an asynchronous task a name using the two overloads, tf::Subflow::async(const std::string& name, F&& f) and tf::Subflow::silent_async(const std::string& name, F&& f). Both methods take an additional argument of a string.
taskflow.emplace([](tf::Subflow& sf){ std::future<void> future = sf.async("name of the task", [](){}); sf.silent_async("another name of the task", [](){}); sf.join(); });
Launch Asynchronous Tasks from a Runtime
The asynchronous tasking feature of tf::
tf::Taskflow taskflow; tf::Executor executor; std::atomic<int> counter{0}; taskflow.emplace([&] (tf::Runtime& rt){ for(int i=0; i<100; i++) { rt.silent_async([&](){ ++counter; })); } rt.corun_all(); // all of the 100 asynchronous tasks will finish by this join assert(counter == 100); }); executor.run(taskflow).wait();
Unlike tf::
tf::Taskflow taskflow; tf::Executor executor; std::atomic<int> counter{0}; taskflow.emplace([&] (tf::Runtime& rt){ // spawn 100 asynchronous tasks and join for(int i=0; i<100; i++) { rt.silent_async([&](){ ++counter; })); } rt.corun_all(); // all of the 100 asynchronous tasks will finish by this join assert(counter == 100); // spawn another 100 asynchronous tasks and join for(int i=0; i<100; i++) { rt.silent_async([&](){ ++counter; })); } rt.corun_all(); // all of the 100 asynchronous tasks will finish by this join assert(counter == 200); }); executor.run(taskflow).wait();
By default, tf::