windows C++-WRL 处理事件

2024-08-26 09:12
文章标签 c++ windows 处理事件 wrl

本文主要是介绍windows C++-WRL 处理事件,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

本文档演示如何使用 Windows 运行时 C++ 模板库 (WRL) 订阅和处理 Windows 运行时对象的事件。

订阅处理事件流程

以下步骤启动对象 ABI::Windows::System::Threading::IDeviceWatcher,并使用事件处理程序监视进度。 通过接口 IDeviceWatcher,可以在添加、删除或更改设备时异步或在后台枚举设备并接收通知。 Callback 函数是此示例的重要组成部分,因为通过它可以指定处理后台操作结果的事件处理程序。 之后提供了完整示例。

尽管通常在通用 Windows 平台应用中使用 Windows 运行时 C++ 模板库,但此示例使用控制台应用进行演示。 通用 Windows 平台应用中不可使用 wprintf_s 等函数。

1. 包括 (#include) 任何所需的 Windows 运行时、C++ 模板库或 C++ 标准库的头文件。

#include <Windows.Devices.Enumeration.h>
#include <wrl/event.h>
#include <stdio.h>using namespace ABI::Windows::Devices::Enumeration;
using namespace ABI::Windows::Foundation;
using namespace Microsoft::WRL;
using namespace Microsoft::WRL::Wrappers;

Windows.Devices.Enumeration.h 声明枚举设备所需的类型。建议在.cpp 文件中使用 using namespace 指令使代码更具可读性。 

2. 声明应用的局部变量。 此示例保留枚举设备和注册令牌的数量计数,以便之后取消订阅事件。:

// Counts the number of enumerated devices.
unsigned int deviceCount = 0;// Event registration tokens that enable us to later unsubscribe from events.
EventRegistrationToken addedToken;
EventRegistrationToken stoppedToken;
EventRegistrationToken enumCompletedToken;

3.  初始化 Windows 运行时库:

// Initialize the Windows Runtime.
RoInitializeWrapper initialize(RO_INIT_MULTITHREADED);
if (FAILED(initialize))
{return PrintError(__LINE__, initialize);
}

 4. 创建一个事件对象,用于向主应用同步枚举过程的完成情况。

// Create an event that is set after device enumeration completes. We later use this event to wait for the timer to complete. 
// This event is for demonstration only in a console app. In most apps, you typically don't wait for async operations to complete.
Event enumerationCompleted(CreateEventEx(nullptr, nullptr, CREATE_EVENT_MANUAL_RESET, WRITE_OWNER | EVENT_ALL_ACCESS));
HRESULT hr = enumerationCompleted.IsValid() ? S_OK : HRESULT_FROM_WIN32(GetLastError());
if (FAILED(hr))
{return PrintError(__LINE__, hr);
}

5. 为 IDeviceWatcher 接口创建激活工厂。 

// Get the activation factory for the IDeviceWatcher interface.
ComPtr<IDeviceInformationStatics> watcherFactory;
hr = ABI::Windows::Foundation::GetActivationFactory(HStringReference(RuntimeClass_Windows_Devices_Enumeration_DeviceInformation).Get(), &watcherFactory);
if (FAILED(hr))
{return PrintError(__LINE__, hr);
}

Windows 运行时使用完全限定名来标识类型。RuntimeClass_Windows_Devices_Enumeration_DeviceInformation 参数是 Windows 运行时提供的字符串,其中包含所需的运行时类名称。 

6. 创建 IDeviceWatcher 对象

// Create a IDeviceWatcher object from the factory.
ComPtr<IDeviceWatcher> watcher;
hr = watcherFactory->CreateWatcher(&watcher);
if (FAILED(hr))
{return PrintError(__LINE__, hr);
}

7. 使用 Callback 函数订阅 AddedEnumerationCompleted 和 Stopped 事件。

// Subscribe to the Added event.
hr = watcher->add_Added(Callback<AddedHandler>([&deviceCount](IDeviceWatcher* watcher, IDeviceInformation*) -> HRESULT
{// Print a message and increment the device count.// When we reach 10 devices, stop enumerating devices.wprintf_s(L"Added device...\n");deviceCount++;if (deviceCount == 10){return watcher->Stop();}return S_OK;}).Get(), &addedToken);
if (FAILED(hr))
{return PrintError(__LINE__, hr);
}hr = watcher->add_Stopped(Callback<StoppedHandler>([=, &enumerationCompleted](IDeviceWatcher* watcher, IInspectable*) -> HRESULT
{wprintf_s(L"Device enumeration stopped.\nRemoving event handlers...");// Unsubscribe from the events. This is shown for demonstration.// The need to remove event handlers depends on the requirements of // your app. For instance, if you only need to handle an event for // a short period of time, you might remove the event handler when you// no longer need it. If you handle an event for the duration of the app,// you might not need to explicitly remove it.HRESULT hr1 = watcher->remove_Added(addedToken);HRESULT hr2 = watcher->remove_Stopped(stoppedToken);HRESULT hr3 = watcher->remove_EnumerationCompleted(enumCompletedToken);// Set the completion event and return.SetEvent(enumerationCompleted.Get());return FAILED(hr1) ? hr1 : FAILED(hr2) ? hr2 : hr3;}).Get(), &stoppedToken);
if (FAILED(hr))
{return PrintError(__LINE__, hr);
}// Subscribe to the EnumerationCompleted event.
hr = watcher->add_EnumerationCompleted(Callback<EnumerationCompletedHandler>([](IDeviceWatcher* watcher, IInspectable*) -> HRESULT
{wprintf_s(L"Enumeration completed.\n");return watcher->Stop();}).Get(), &enumCompletedToken);
if (FAILED(hr))
{return PrintError(__LINE__, hr);
}

8. 启动枚举过程。

wprintf_s(L"Starting device enumeration...\n");
hr = watcher->Start();
if (FAILED(hr))
{return PrintError(__LINE__, hr);
}

9. 等待枚举过程完成,然后输出消息。 所有 ComPtr 和 RAII 对象都离开范围并自动释放。 

// Wait for the operation to complete.
WaitForSingleObjectEx(enumerationCompleted.Get(), INFINITE, FALSE);wprintf_s(L"Enumerated %u devices.\n", deviceCount);// All smart pointers and RAII objects go out of scope here.
完整代码:
// wrl-consume-events.cpp
// compile with: runtimeobject.lib
#include <Windows.Devices.Enumeration.h>
#include <wrl/event.h>
#include <stdio.h>using namespace ABI::Windows::Devices::Enumeration;
using namespace ABI::Windows::Foundation;
using namespace Microsoft::WRL;
using namespace Microsoft::WRL::Wrappers;// Prints an error string for the provided source code line and HRESULT
// value and returns the HRESULT value as an int.
int PrintError(unsigned int line, HRESULT hr)
{wprintf_s(L"ERROR: Line:%d HRESULT: 0x%X\n", line, hr);return hr;
}int wmain()
{// Type define the event handler types to make the code more readable.typedef __FITypedEventHandler_2_Windows__CDevices__CEnumeration__CDeviceWatcher_Windows__CDevices__CEnumeration__CDeviceInformation AddedHandler;typedef __FITypedEventHandler_2_Windows__CDevices__CEnumeration__CDeviceWatcher_IInspectable EnumerationCompletedHandler;typedef __FITypedEventHandler_2_Windows__CDevices__CEnumeration__CDeviceWatcher_IInspectable StoppedHandler;// Counts the number of enumerated devices.unsigned int deviceCount = 0;// Event registration tokens that enable us to later unsubscribe from events.EventRegistrationToken addedToken;EventRegistrationToken stoppedToken;EventRegistrationToken enumCompletedToken;// Initialize the Windows Runtime.RoInitializeWrapper initialize(RO_INIT_MULTITHREADED);if (FAILED(initialize)){return PrintError(__LINE__, initialize);}// Create an event that is set after device enumeration completes. We later use this event to wait for the timer to complete. // This event is for demonstration only in a console app. In most apps, you typically don't wait for async operations to complete.Event enumerationCompleted(CreateEventEx(nullptr, nullptr, CREATE_EVENT_MANUAL_RESET, WRITE_OWNER | EVENT_ALL_ACCESS));HRESULT hr = enumerationCompleted.IsValid() ? S_OK : HRESULT_FROM_WIN32(GetLastError());if (FAILED(hr)){return PrintError(__LINE__, hr);}// Get the activation factory for the IDeviceWatcher interface.ComPtr<IDeviceInformationStatics> watcherFactory;hr = ABI::Windows::Foundation::GetActivationFactory(HStringReference(RuntimeClass_Windows_Devices_Enumeration_DeviceInformation).Get(), &watcherFactory);if (FAILED(hr)){return PrintError(__LINE__, hr);}// Create a IDeviceWatcher object from the factory.ComPtr<IDeviceWatcher> watcher;hr = watcherFactory->CreateWatcher(&watcher);if (FAILED(hr)){return PrintError(__LINE__, hr);}// Subscribe to the Added event.hr = watcher->add_Added(Callback<AddedHandler>([&deviceCount](IDeviceWatcher* watcher, IDeviceInformation*) -> HRESULT{// Print a message and increment the device count.// When we reach 10 devices, stop enumerating devices.wprintf_s(L"Added device...\n");deviceCount++;if (deviceCount == 10){return watcher->Stop();}return S_OK;}).Get(), &addedToken);if (FAILED(hr)){return PrintError(__LINE__, hr);}hr = watcher->add_Stopped(Callback<StoppedHandler>([=, &enumerationCompleted](IDeviceWatcher* watcher, IInspectable*) -> HRESULT{wprintf_s(L"Device enumeration stopped.\nRemoving event handlers...");// Unsubscribe from the events. This is shown for demonstration.// The need to remove event handlers depends on the requirements of // your app. For instance, if you only need to handle an event for // a short period of time, you might remove the event handler when you// no longer need it. If you handle an event for the duration of the app,// you might not need to explicitly remove it.HRESULT hr1 = watcher->remove_Added(addedToken);HRESULT hr2 = watcher->remove_Stopped(stoppedToken);HRESULT hr3 = watcher->remove_EnumerationCompleted(enumCompletedToken);// Set the completion event and return.SetEvent(enumerationCompleted.Get());return FAILED(hr1) ? hr1 : FAILED(hr2) ? hr2 : hr3;}).Get(), &stoppedToken);if (FAILED(hr)){return PrintError(__LINE__, hr);}// Subscribe to the EnumerationCompleted event.hr = watcher->add_EnumerationCompleted(Callback<EnumerationCompletedHandler>([](IDeviceWatcher* watcher, IInspectable*) -> HRESULT{wprintf_s(L"Enumeration completed.\n");return watcher->Stop();}).Get(), &enumCompletedToken);if (FAILED(hr)){return PrintError(__LINE__, hr);}wprintf_s(L"Starting device enumeration...\n");hr = watcher->Start();if (FAILED(hr)){return PrintError(__LINE__, hr);}// Wait for the operation to complete.WaitForSingleObjectEx(enumerationCompleted.Get(), INFINITE, FALSE);wprintf_s(L"Enumerated %u devices.\n", deviceCount);// All smart pointers and RAII objects go out of scope here.
}
/*
Sample output:
Starting device enumeration...
Added device...
Added device...
Added device...
Added device...
Added device...
Added device...
Added device...
Added device...
Added device...
Added device...
Device enumeration stopped.
Removing event handlers...
Enumerated 10 devices.
*/

这篇关于windows C++-WRL 处理事件的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1108149

相关文章

Java调用C++动态库超详细步骤讲解(附源码)

《Java调用C++动态库超详细步骤讲解(附源码)》C语言因其高效和接近硬件的特性,时常会被用在性能要求较高或者需要直接操作硬件的场合,:本文主要介绍Java调用C++动态库的相关资料,文中通过代... 目录一、直接调用C++库第一步:动态库生成(vs2017+qt5.12.10)第二步:Java调用C++

C/C++错误信息处理的常见方法及函数

《C/C++错误信息处理的常见方法及函数》C/C++是两种广泛使用的编程语言,特别是在系统编程、嵌入式开发以及高性能计算领域,:本文主要介绍C/C++错误信息处理的常见方法及函数,文中通过代码介绍... 目录前言1. errno 和 perror()示例:2. strerror()示例:3. perror(

C++变换迭代器使用方法小结

《C++变换迭代器使用方法小结》本文主要介绍了C++变换迭代器使用方法小结,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录1、源码2、代码解析代码解析:transform_iterator1. transform_iterat

详解C++中类的大小决定因数

《详解C++中类的大小决定因数》类的大小受多个因素影响,主要包括成员变量、对齐方式、继承关系、虚函数表等,下面就来介绍一下,具有一定的参考价值,感兴趣的可以了解一下... 目录1. 非静态数据成员示例:2. 数据对齐(Padding)示例:3. 虚函数(vtable 指针)示例:4. 继承普通继承虚继承5.

C++中std::distance使用方法示例

《C++中std::distance使用方法示例》std::distance是C++标准库中的一个函数,用于计算两个迭代器之间的距离,本文主要介绍了C++中std::distance使用方法示例,具... 目录语法使用方式解释示例输出:其他说明:总结std::distance&n编程bsp;是 C++ 标准

C++ 中的 if-constexpr语法和作用

《C++中的if-constexpr语法和作用》if-constexpr语法是C++17引入的新语法特性,也被称为常量if表达式或静态if(staticif),:本文主要介绍C++中的if-c... 目录1 if-constexpr 语法1.1 基本语法1.2 扩展说明1.2.1 条件表达式1.2.2 fa

C++中::SHCreateDirectoryEx函数使用方法

《C++中::SHCreateDirectoryEx函数使用方法》::SHCreateDirectoryEx用于创建多级目录,类似于mkdir-p命令,本文主要介绍了C++中::SHCreateDir... 目录1. 函数原型与依赖项2. 基本使用示例示例 1:创建单层目录示例 2:创建多级目录3. 关键注

C++从序列容器中删除元素的四种方法

《C++从序列容器中删除元素的四种方法》删除元素的方法在序列容器和关联容器之间是非常不同的,在序列容器中,vector和string是最常用的,但这里也会介绍deque和list以供全面了解,尽管在一... 目录一、简介二、移除给定位置的元素三、移除与某个值相等的元素3.1、序列容器vector、deque

C++常见容器获取头元素的方法大全

《C++常见容器获取头元素的方法大全》在C++编程中,容器是存储和管理数据集合的重要工具,不同的容器提供了不同的接口来访问和操作其中的元素,获取容器的头元素(即第一个元素)是常见的操作之一,本文将详细... 目录一、std::vector二、std::list三、std::deque四、std::forwa

C++字符串提取和分割的多种方法

《C++字符串提取和分割的多种方法》在C++编程中,字符串处理是一个常见的任务,尤其是在需要从字符串中提取特定数据时,本文将详细探讨如何使用C++标准库中的工具来提取和分割字符串,并分析不同方法的适用... 目录1. 字符串提取的基本方法1.1 使用 std::istringstream 和 >> 操作符示