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

相关文章

使用Python创建一个功能完整的Windows风格计算器程序

《使用Python创建一个功能完整的Windows风格计算器程序》:本文主要介绍如何使用Python和Tkinter创建一个功能完整的Windows风格计算器程序,包括基本运算、高级科学计算(如三... 目录python实现Windows系统计算器程序(含高级功能)1. 使用Tkinter实现基础计算器2.

Windows系统宽带限制如何解除?

《Windows系统宽带限制如何解除?》有不少用户反映电脑网速慢得情况,可能是宽带速度被限制的原因,只需解除限制即可,具体该如何操作呢?本文就跟大家一起来看看Windows系统解除网络限制的操作方法吧... 有不少用户反映电脑网速慢得情况,可能是宽带速度被限制的原因,只需解除限制即可,具体该如何操作呢?本文

windows和Linux使用命令行计算文件的MD5值

《windows和Linux使用命令行计算文件的MD5值》在Windows和Linux系统中,您可以使用命令行(终端或命令提示符)来计算文件的MD5值,文章介绍了在Windows和Linux/macO... 目录在Windows上:在linux或MACOS上:总结在Windows上:可以使用certuti

C#如何调用C++库

《C#如何调用C++库》:本文主要介绍C#如何调用C++库方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录方法一:使用P/Invoke1. 导出C++函数2. 定义P/Invoke签名3. 调用C++函数方法二:使用C++/CLI作为桥接1. 创建C++/CL

Windows 上如果忘记了 MySQL 密码 重置密码的两种方法

《Windows上如果忘记了MySQL密码重置密码的两种方法》:本文主要介绍Windows上如果忘记了MySQL密码重置密码的两种方法,本文通过两种方法结合实例代码给大家介绍的非常详细,感... 目录方法 1:以跳过权限验证模式启动 mysql 并重置密码方法 2:使用 my.ini 文件的临时配置在 Wi

C++如何通过Qt反射机制实现数据类序列化

《C++如何通过Qt反射机制实现数据类序列化》在C++工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作,所以本文就来聊聊C++如何通过Qt反射机制实现数据类序列化吧... 目录设计预期设计思路代码实现使用方法在 C++ 工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作。由于数据类

Windows Docker端口占用错误及解决方案总结

《WindowsDocker端口占用错误及解决方案总结》在Windows环境下使用Docker容器时,端口占用错误是开发和运维中常见且棘手的问题,本文将深入剖析该问题的成因,介绍如何通过查看端口分配... 目录引言Windows docker 端口占用错误及解决方案汇总端口冲突形成原因解析诊断当前端口情况解

Linux下如何使用C++获取硬件信息

《Linux下如何使用C++获取硬件信息》这篇文章主要为大家详细介绍了如何使用C++实现获取CPU,主板,磁盘,BIOS信息等硬件信息,文中的示例代码讲解详细,感兴趣的小伙伴可以了解下... 目录方法获取CPU信息:读取"/proc/cpuinfo"文件获取磁盘信息:读取"/proc/diskstats"文

C++使用printf语句实现进制转换的示例代码

《C++使用printf语句实现进制转换的示例代码》在C语言中,printf函数可以直接实现部分进制转换功能,通过格式说明符(formatspecifier)快速输出不同进制的数值,下面给大家分享C+... 目录一、printf 原生支持的进制转换1. 十进制、八进制、十六进制转换2. 显示进制前缀3. 指

C++中初始化二维数组的几种常见方法

《C++中初始化二维数组的几种常见方法》本文详细介绍了在C++中初始化二维数组的不同方式,包括静态初始化、循环、全部为零、部分初始化、std::array和std::vector,以及std::vec... 目录1. 静态初始化2. 使用循环初始化3. 全部初始化为零4. 部分初始化5. 使用 std::a