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++判断水仙花数并输出示例代码

《利用c++判断水仙花数并输出示例代码》水仙花数是指一个三位数,其各位数字的立方和恰好等于该数本身,:本文主要介绍利用c++判断水仙花数并输出的相关资料,文中通过代码介绍的非常详细,需要的朋友可以... 以下是使用C++实现的相同逻辑代码:#include <IOStream>#include <vec

基于C++的UDP网络通信系统设计与实现详解

《基于C++的UDP网络通信系统设计与实现详解》在网络编程领域,UDP作为一种无连接的传输层协议,以其高效、低延迟的特性在实时性要求高的应用场景中占据重要地位,下面我们就来看看如何从零开始构建一个完整... 目录前言一、UDP服务器UdpServer.hpp1.1 基本框架设计1.2 初始化函数Init详解

C++ 右值引用(rvalue references)与移动语义(move semantics)深度解析

《C++右值引用(rvaluereferences)与移动语义(movesemantics)深度解析》文章主要介绍了C++右值引用和移动语义的设计动机、基本概念、实现方式以及在实际编程中的应用,... 目录一、右值引用(rvalue references)与移动语义(move semantics)设计动机1

windows下安装Nginx全过程

《windows下安装Nginx全过程》文章介绍了HTTP和反向代理服务器的概念,包括正向代理和反向代理的区别,并详细描述了如何安装和配置Nginx作为反向代理服务器... 目录概念代理正向代理反向代理安装基本属性nginx.conf查询结构属性使用运行重启停止总结概念是一个高性能的HTTP和反向代理we

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