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

相关文章

C++ move 的作用详解及陷阱最佳实践

《C++move的作用详解及陷阱最佳实践》文章详细介绍了C++中的`std::move`函数的作用,包括为什么需要它、它的本质、典型使用场景、以及一些常见陷阱和最佳实践,感兴趣的朋友跟随小编一起看... 目录C++ move 的作用详解一、一句话总结二、为什么需要 move?C++98/03 的痛点⚡C++

详解C++ 存储二进制数据容器的几种方法

《详解C++存储二进制数据容器的几种方法》本文主要介绍了详解C++存储二进制数据容器,包括std::vector、std::array、std::string、std::bitset和std::ve... 目录1.std::vector<uint8_t>(最常用)特点:适用场景:示例:2.std::arra

C++构造函数中explicit详解

《C++构造函数中explicit详解》explicit关键字用于修饰单参数构造函数或可以看作单参数的构造函数,阻止编译器进行隐式类型转换或拷贝初始化,本文就来介绍explicit的使用,感兴趣的可以... 目录1. 什么是explicit2. 隐式转换的问题3.explicit的使用示例基本用法多参数构造

C++,C#,Rust,Go,Java,Python,JavaScript的性能对比全面讲解

《C++,C#,Rust,Go,Java,Python,JavaScript的性能对比全面讲解》:本文主要介绍C++,C#,Rust,Go,Java,Python,JavaScript性能对比全面... 目录编程语言性能对比、核心优势与最佳使用场景性能对比表格C++C#RustGoJavapythonjav

C++打印 vector的几种方法小结

《C++打印vector的几种方法小结》本文介绍了C++中遍历vector的几种方法,包括使用迭代器、auto关键字、typedef、计数器以及C++11引入的范围基础循环,具有一定的参考价值,感兴... 目录1. 使用迭代器2. 使用 auto (C++11) / typedef / type alias

C++ scoped_ptr 和 unique_ptr对比分析

《C++scoped_ptr和unique_ptr对比分析》本文介绍了C++中的`scoped_ptr`和`unique_ptr`,详细比较了它们的特性、使用场景以及现代C++推荐的使用`uni... 目录1. scoped_ptr基本特性主要特点2. unique_ptr基本用法3. 主要区别对比4. u

C++11中的包装器实战案例

《C++11中的包装器实战案例》本文给大家介绍C++11中的包装器实战案例,本文结合实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录引言1.std::function1.1.什么是std::function1.2.核心用法1.2.1.包装普通函数1.2.

C++多线程开发环境配置方法

《C++多线程开发环境配置方法》文章详细介绍了如何在Windows上安装MinGW-w64和VSCode,并配置环境变量和编译任务,使用VSCode创建一个C++多线程测试项目,并通过配置tasks.... 目录下载安装 MinGW-w64下载安装VS code创建测试项目配置编译任务创建 tasks.js

在C#中调用Windows防火墙界面的常见方式

《在C#中调用Windows防火墙界面的常见方式》在C#中调用Windows防火墙界面(基础设置或高级安全设置),可以使用进程启动(Process.Start)或Win32API来实现,所以本文给大家... 目录引言1. 直接启动防火墙界面(1) 打开基本防火墙设置(firewall.cpl)(2) 打开高

基于Python实现局域网内Windows桌面文件传输

《基于Python实现局域网内Windows桌面文件传输》这篇文章介绍了如何使用Python实现一个局域网文件传输系统,包括发送端和接收端的代码示例,发送端和接收端都需要在同一局域网内运行,并且确保防... 目录发送端代码 (sender.py)接收端代码 (receiver.py)图形界面版本 (可选)使