注册取消回调
Windows 运行时的取消不会自动流向其他异步对象。 但是,可以注册取消回调,这是 Windows SDK 版本 10.0.17763.0(Windows 10 版本 1809)中引入的功能。 这是一种先发性的挂钩,可据此传播取消,以及与现有的并发库集成。
在以下代码示例中,NestedCoroutineAsync 将执行工作,但其中不包含特殊的取消逻辑。 CancelationPropagatorAsync 实质上是协同例程中的一个包装器;该包装器提前转发取消。
// main.cpp
#include <iostream>
#include <winrt/Windows.Foundation.h>
using namespace winrt;
using namespace Windows::Foundation;
using namespace std::chrono_literals;
IAsyncAction NestedCoroutineAsync()
{
while (true)
{
std::cout << "NestedCoroutineAsync: do some work for 1 second" << std::endl;
co_await 1s;
}
}
IAsyncAction CancelationPropagatorAsync()
{
auto cancelation_token{ co_await winrt::get_cancellation_token() };
auto nested_coroutine{ NestedCoroutineAsync() };
cancelation_token.callback([=]
{
nested_coroutine.Cancel();
});
co_await nested_coroutine;
}
IAsyncAction MainCoroutineAsync()
{
auto cancelation_propagator{ CancelationPropagatorAsync() };
co_await 3s;
cancelation_propagator.Cancel();
}
int main()
{
winrt::init_apartment();
MainCoroutineAsync().get();
}
CancelationPropagatorAsync 为自身的取消回调注册一个 lambda 函数,然后等待(此时会暂停)嵌套的工作完成。 如果 CancellationPropagatorAsync 已取消,则会将取消传播到嵌套的协同例程。 无需轮询取消;取消不会无限期阻塞。 此机制足够灵活,使用它可以与完全不了解 C++/WinRT 的协同例程或并发库互操作。
报告进度
如果协同例程返回 IAsyncActionWithProgress 或 IAsyncOperationWithProgress,则你可以检索 winrt::get_progress_token 函数返回的对象,并使用该对象将进度报告给进度处理程序。 下面是代码示例。
// main.cpp
#include <iostream>
#include <winrt/Windows.Foundation.h>
using namespace winrt;
using namespace Windows::Foundation;
using namespace std::chrono_literals;
IAsyncOperationWithProgress<double, double> CalcPiTo5DPs()
{
auto progress{ co_await winrt::get_progress_token() };
co_await 1s;
double pi_so_far{ 3.1 };
progress.set_result(pi_so_far);
progress(0.2);
co_await 1s;
pi_so_far += 4.e-2;
progress.set_result(pi_so_far);
progress(0.4);
co_await 1s;
pi_so_far += 1.e-3;
progress.set_result(pi_so_far);
progress(0.6);
co_await 1s;
pi_so_far += 5.e-4;
progress.set_result(pi_so_far);
progress(0.8);
co_await 1s;
pi_so_far += 9.e-5;
progress.set_result(pi_so_far);
progress(1.0);
co_return pi_so_far;
}
IAsyncAction DoMath()
{
auto async_op_with_progress{ CalcPiTo5DPs() };
async_op_with_progress.Progress([](auto const& sender, double progress)
{
std::wcout << L"CalcPiTo5DPs() reports progress: " << progress << L". "
<< L"Value so far: " << sender.GetResults() << std::endl;
});
double pi{ co_await async_op_with_progress };
std::wcout << L"CalcPiTo5DPs() is complete !" << std::endl;
std::wcout << L"Pi is approx.: " << pi << std::endl;
}
int main()
{
winrt::init_apartment();
DoMath().get();
}
若要报告进度,请调用以进度值作为参数的进度标记。 若要设置临时结果,请对进度标记使用 set_result() 方法。
报告临时结果需要 C++/WinRT 版本 2.0.210309.3 或更高版本。上面的示例选择为每个进度报告都设置一个临时结果。 可以选择随时报告临时结果(如果有)。 它不需要加上进度报告。
对一个异步操作或运算实现多个完成处理程序是错误的做法。 可对其已完成的事件使用单个委托,或者可对其运行 co_await。 如果同时采用这两种方法,则第二种方法会失败。 以下两种完成处理程序都是适当的;但不能同时对同一个异步对象使用两者。
auto async_op_with_progress{ CalcPiTo5DPs() };
async_op_with_progress.Completed([](auto const& sender, AsyncStatus /* status */)
{
double pi{ sender.GetResults() };
});
auto async_op_with_progress{ CalcPiTo5DPs() };
double pi{ co_await async_op_with_progress };
发后不理
有时,某个任务可与其他工作并发执行,你无需等待该任务完成(因为没有其他工作依赖于它),也无需该任务返回值。 在这种情况下,可以激发该任务,然后忘记它。 为此,可以编写返回类型为 winrt::fire_and_forget(而不是某个 Windows 运行时异步操作类型,或 concurrency:: task)的协同例程。
// main.cpp
#include <winrt/Windows.Foundation.h>
using namespace winrt;
using namespace std::chrono_literals;
winrt::fire_and_forget CompleteInFiveSeconds()
{
co_await 5s;
}
int main()
{
winrt::init_apartment();
CompleteInFiveSeconds();
// Do other work here.
}
winrt::fire_and_forget 也可用作事件处理程序的返回类型,前提是需在其中执行异步操作。 下面是一个示例。
winrt::fire_and_forget MyClass::MyMediaBinder_OnBinding(MediaBinder const&, MediaBindingEventArgs args)
{
auto lifetime{ get_strong() }; // Prevent *this* from prematurely being destructed.
auto ensure_completion{ unique_deferral(args.GetDeferral()) }; // Take a deferral, and ensure that we complete it.
auto file{ co_await StorageFile::GetFileFromApplicationUriAsync(Uri(L"ms-appx:///video_file.mp4")) };
args.SetStorageFile(file);
// The destructor of unique_deferral completes the deferral here.
}
第一个参数 (sender) 未命名,因为我们从未使用它。 因此,可以安全地将它保留为引用。 但请注意,args 是按值传递的。