当前位置: 首页 > news >正文

windows C++-在 C++/WinRT 中使用委托处理事件(下)

撤销已注册的委托

当你注册委托时,通常会向你返回一个令牌。 随后,可以使用该令牌撤销委托;这意味着将从事件取消注册委托,再次引发该事件时不会调用该委托。

为简单起见,上面的代码示例都没有介绍如何执行该操作。 但下面这个代码示例将令牌存储在结构的专用数据成员中,并在析构函数中撤销令牌的处理程序。

struct Example : ExampleT<Example>
{Example(winrt::Windows::UI::Xaml::Controls::Button const& button) : m_button(button){m_token = m_button.Click([this](IInspectable const&, RoutedEventArgs const&){// ...});}~Example(){m_button.Click(m_token);}private:winrt::Windows::UI::Xaml::Controls::Button m_button;winrt::event_token m_token;
};

可以不进行上面示例中的强引用,而存储对按钮的弱引用。

当事件源以同步方式引发其事件时,你就可以放心地撤销处理程序:不会收到更多事件了。 但对于异步事件,即使在撤销(尤其是在析构函数中撤销)后,你的对象在开始析构后仍可能收到正在进行的事件。 在析构之前找到取消订阅的地方也许可以缓解此问题。

或者,当你注册代理时,也可以指定 winrt::auto_revoke(即 winrt::auto_revoke_t 类型的值)以请求一个事件撤销程序(winrt::event_revoker 类型) 。 事件撤销程序为你保留对事件源(引发事件的对象)的弱引用。 可以通过调用 event_revoker::revoke 成员函数手动撤销;但事件撤销程序会在该函数超出范围时自动调用函数本身 。 撤销函数检查事件源是否仍然存在,如果存在,将撤销你的代理 。 在本示例中,无需存储事件源,并且不需要析构函数。

struct Example : ExampleT<Example>
{Example(winrt::Windows::UI::Xaml::Controls::Button button){m_event_revoker = button.Click(winrt::auto_revoke,[this](IInspectable const& /* sender */,RoutedEventArgs const& /* args */){// ...});}private:winrt::Windows::UI::Xaml::Controls::Button::Click_revoker m_event_revoker;
};

 下面是摘自 ButtonBase::Click 事件的文档主题的语法块 。 它显示了三个不同的注册和撤销函数。 可以清楚地看到从第三个重载进行声明时需要哪种类型的事件撤销程序。 你可以将相同类型的委托传递给“注册”和“使用 event_revoker 撤销”重载 。

// Register
winrt::event_token Click(winrt::Windows::UI::Xaml::RoutedEventHandler const& handler) const;// Revoke with event_token
void Click(winrt::event_token const& token) const;// Revoke with event_revoker
Button::Click_revoker Click(winrt::auto_revoke_t,winrt::Windows::UI::Xaml::RoutedEventHandler const& handler) const;

 在上述代码示例中,Button::Click_revoker 是 winrt::event_revoker<winrt::Windows::UI::Xaml::Controls::Primitives::IButtonBase> 的类型别名。 类似的模式适用于所有 C++/WinRT 事件。 每个 Windows 运行时事件都具有返回事件撤销程序的撤销函数重载,且该撤销程序的类型是事件源的成员。

另一个示例是,CoreWindow::SizeChanged 事件具有注册函数重载,它返回类型 CoreWindow::SizeChanged_revoker 的值 。

在页面-导航方案中,可以考虑撤销处理程序。 如果反复进入某个页面然后退出,则可以在离开该页面时撤销任何处理程序。 或者,如果你重复使用同一页面实例,请检查令牌的值,仅在该值未设置时注册 (if (!m_token){ ... })。 第三个选项是将事件撤销程序作为数据成员存储在页面中。 第四个选项(将在本主题后面描述)是捕获对 lambda 函数中的 this 对象的强引用或弱引用 。 

 如果“自动撤销”委托无法注册

如果在注册委托时尝试指定 winrt::auto_revoke,并且结果是 winrt::hresult_no_interface 异常,那么这通常意味着,事件源不支持弱引用。 例如,这是 Windows.UI.Composition 命名空间中的常见情况。 在此情况下,不能使用自动撤销功能。 必须故障回复到手动撤销事件处理程序。

异步操作和运算的委托类型

前面的示例使用的是 RoutedEventHandler 委托类型,但当然还有很多其他委托类型 。 例如,异步操作和运算(带进度和不带进度)具有期望相应类型的委托的已完成和/或进度事件。 例如,带进度的异步运算进度事件(可以是实现 IAsyncOperationWithProgress 的任何内容)需要 AsyncOperationProgressHandler 类型的委托 。 下面是使用 lambda 函数创作该类型的委托的代码示例。 该示例还演示了如何创作 AsyncOperationWithProgressCompletedHandler 代理 。

#include <winrt/Windows.Foundation.h>
#include <winrt/Windows.Web.Syndication.h>using namespace winrt;
using namespace Windows::Foundation;
using namespace Windows::Web::Syndication;void ProcessFeedAsync()
{Uri rssFeedUri{ L"https://blogs.windows.com/feed" };SyndicationClient syndicationClient;auto async_op_with_progress = syndicationClient.RetrieveFeedAsync(rssFeedUri);async_op_with_progress.Progress([](IAsyncOperationWithProgress<SyndicationFeed,RetrievalProgress> const& /* sender */,RetrievalProgress const& args){uint32_t bytes_retrieved = args.BytesRetrieved;// use bytes_retrieved;});async_op_with_progress.Completed([](IAsyncOperationWithProgress<SyndicationFeed,RetrievalProgress> const& sender,AsyncStatus const /* asyncStatus */){SyndicationFeed syndicationFeed = sender.GetResults();// use syndicationFeed;});// or (but this function must then be a coroutine, and return IAsyncAction)// SyndicationFeed syndicationFeed{ co_await async_op_with_progress };
}

如上面的“协同程序”注释所示,与将代理与异步操作和运算的已完成事件结合使用相比,你可能会发现使用协同程序更自然。 

对一个异步操作或运算实现多个完成处理程序是错误的做法 。 可对其已完成的事件使用单个委托,或者可对其运行 co_await。 如果同时采用这两种方法,则第二种方法会失败。

如果坚持使用委托而不是协同程序,则可以选择更简单的语法。

async_op_with_progress.Completed([](auto&& /*sender*/, AsyncStatus const /* args */)
{// ...
});

 返回一个值的代理类型

某些委托类型本身必须返回一个值。 示例:ListViewItemToKeyHandler,它将返回一个字符串 。 下面是创作该类型的委托的示例(请注意,lambda 函数将返回一个值)。

using namespace winrt::Windows::UI::Xaml::Controls;winrt::hstring f(ListView listview)
{return ListViewPersistenceHelper::GetRelativeScrollPosition(listview, [](IInspectable const& item){return L"key for item goes here";});
}

使用事件处理委托安全访问 this 指针

如果你使用对象的成员函数处理事件,或者从对象成员函数中的某个 lambda 函数内部处理事件,则需要考虑事件接收方(处理事件的对象)和事件源(引发事件的对象)的相对生存期。

 

相关文章:

windows C++-在 C++/WinRT 中使用委托处理事件(下)

撤销已注册的委托 当你注册委托时&#xff0c;通常会向你返回一个令牌。 随后&#xff0c;可以使用该令牌撤销委托&#xff1b;这意味着将从事件取消注册委托&#xff0c;再次引发该事件时不会调用该委托。 为简单起见&#xff0c;上面的代码示例都没有介绍如何执行该操作。 …...

【实用工具】Stirling-PDF: 优质开源的PDF处理工具/编辑工具-含入门安装教程

文章目录 项目简介功能展示Page Operations 页面操作Conversion Operations 转换操作Security & Permissions 安全与权限Other Operations 其他业务 如何安装并使用Docker RunDocker Compose 项目简介 这是一款使用 Docker 的基于本地托管网络的强大 PDF 操作工具。它能让…...

opencv 深度图视差图可视化案例

参考:https://www.cnblogs.com/zyly/p/9373991.html(图片这里面下载的) https://blog.csdn.net/He3he3he/article/details/101053457 原理 双目摄像头 视差公式: 三角形对应推算 深度距离转换: 这里d是视差Disparity 代码 下面两种计算视差方法: import os impor…...

Golang | Leetcode Golang题解之第330题按要求补齐数组

题目&#xff1a; 题解&#xff1a; func minPatches(nums []int, n int) (patches int) {for i, x : 0, 1; x < n; {if i < len(nums) && nums[i] < x {x nums[i]i} else {x * 2patches}}return }...

算法训练(leetcode)第五十二天 | Bellman_ford 队列优化算法(SPFA)、BF算法判断负回路、BF之单源有限最短路(有负回路)

刷题记录 94. 城市间货物运输 I-Bellman_ford 队列优化算法&#xff08;SPFA&#xff09;95. 城市间货物运输 II-BF算法判断负回路96. 城市间货物运输 III-BF之单源有限最短路(有负回路) 94. 城市间货物运输 I-Bellman_ford 队列优化算法&#xff08;SPFA&#xff09; 题目地址…...

SpringBoot中整合RabbitMQ(测试+部署上线 最完整)

一、RabbitMQ安装 由于在测试环境中&#xff0c;我们现在虚拟机上基于docker安装mq docker run \-e RABBITMQ_DEFAULT_USERquick \-e RABBITMQ_DEFAULT_PASS123 \-v mq-plugins:/plugins \--name mq \--hostname mq \-p 15672:15672 \-p 5672:5672 \--network your-net\-d \r…...

算法板子:线性DP——算出三角形中的最大路径值、求最长上升子序列、求最长公共子序列

目录 一、数字三角形——算出三角形中的最大路径值 二、最长上升子序列——求一个数组中的最长递增子序列 三、最长公共子序列——求两个字符串中的最长公共子序列 一、数字三角形——算出三角形中的最大路径值 #include <iostream> using namespace std;const int N …...

【C++】值传递

函数值传递的特点&#xff1a;值传递过程中即使形参改变也不会改变实参 没有返回值的函数用“ void ”定义 下面是一个实例&#xff1a; #include<iostream> using namespace std;//值传递 //定义函数&#xff0c;实现两个数字进行交换函数//如果函数不需要返回值&…...

工业三防平板助力MES系统打造工厂移动式生产管理

随着工业4.0时代的到来&#xff0c;智能制造、数字化车间等概念层出不穷&#xff0c;生产过程的可视化管理也成为了企业提升效率、优化生产的关键。而工业三防平板&#xff0c;凭借其坚固耐用、功能强大、便携易用等特性&#xff0c;成为了实现生产过程可视化管理的重要利器&am…...

keepalived+nginx实现的简单高可用故障转移

keepalived和nginx和适配 nginx服务停止后对keepalived的影响最近研究了一下keepalived绑定虚拟Ip,然后实现集群的方案,发现实现故障转移的模式,只有在keepalived服务整个挂掉后才能实现虚拟IP的漂移,和实际应用的场景不怎么适配,所以把它和nginx结合在一起实现集群高可用…...

openai api使用

1OpenAI 的 API 介绍 1.1 api分类 常用的 OpenAI Api 接口总共分为 4 类&#xff1a;对话类、私有化模型训练类、通用类、图片 & 音频类&#xff0c;其中对话类与私有化模型训练类是最常用的。 a .对话类 这类是最常用也是最核心的接口&#xff0c;用于人机对话。对话类…...

带你走进haproxy的世界

华子目录 前言什么是负载均衡为什么用haproxy负载均衡负载均衡公司负载均衡类型四层负载均衡七层负载均衡四层和七层的区别 haproxy介绍haproxy的安装与服务信息软件安装haproxy基本配置信息proxies配置socat工具 haproxy算法静态算法动态算法其他算法 高级功能及配置基于cooki…...

STM32--中断使用(超详细!)

STM32中断机制是嵌入式系统设计中一个非常重要的组成部分&#xff0c;它允许单片机在执行程序的过程中&#xff0c;对外部或内部发生的事件做出快速响应。以下是一篇关于STM32中断机制的详细介绍和示例代码&#xff0c;希望能够帮助你更好地理解和应用中断。 一、中断的基本概…...

【深度学习实践】基于深度学习的图像去雾算法-ChaIR-实践

本文介绍一个去雾算法ChaIR的使用方法&#xff0c;可以完成图像去雾&#xff0c;也可以用于图像去雨、去噪音等任务。本文不涉及论文原理&#xff0c;只包含源代码的跑通和使用。 先展示一下效果&#xff1a; 原图去雾 论文&#xff1a;Exploring the potential of channel …...

《乳腺密度高的女性中,使用AI辅助的乳腺X线筛查与补充筛查超声的比较研究》| 文献速递-基于深度学习的乳房、前列腺疾病诊断系统

Title 题目 Screening Outcomes of Mammography with AI in Dense Breasts: A Comparative Study with Supplemental Screening US 《乳腺密度高的女性中&#xff0c;使用AI辅助的乳腺X线筛查与补充筛查超声的比较研究》 Background 背景 Comparative performance between…...

crm 销售管理系统有哪些?国内外排名前十盘点

本文深入对比的 crm销售管理系统有&#xff1a;1.纷享销客&#xff1b; 2.Zoho CRM&#xff1b; 3.销售易&#xff1b; 4.有赞CRM&#xff1b; 5.Salesforce&#xff1b; 6.HubSpot&#xff1b; 7.简道云CRM&#xff1b; 8.爱客CRM&#xff1b; 9.Apptivo。 如果你正寻找一种方…...

package-lock.json 要提交到git吗?

之前一直没有提交package-lock.json文件到git仓库&#xff0c;直到我打包失败了。。。 我才知道package-lock.json需要提交到‌git仓库。 ‌ npm官网建议将package-lock.json一起提交到代码库中&#xff0c;不要忽略它。‌ package-lock.json的主要作用是锁定dependencies的版…...

算法学习day32

一、解码方法II&#xff08;解码方法I的升级版&#xff09; 在I的基础上增加了*&#xff0c;可以代替1-9中任意一个数字&#xff0c;求解码的方法有多少种 输入&#xff1a;s "*" 输出&#xff1a;9 解释&#xff1a;这一条编码消息可以表示 "1"、"…...

知识与智慧

前两天在medium上看到一篇文章&#xff0c;探讨知识&#xff08;knowledge&#xff09;和智慧&#xff08;wisdom&#xff09;之间的区别&#xff0c;很受启发&#xff0c;结合自己的经历和理解&#xff0c;形成此文&#xff1a; 何为知识 知识通常指的是信息的积累和对特定领…...

使用FFmpeg实现摄像头RTMP实时推流

在当今的数字时代,视频直播已成为连接人与人之间的重要桥梁,广泛应用于在线教育、远程会议、娱乐直播等多个领域。随着技术的不断进步,人们对于直播的实时性、稳定性和高质量需求日益增加。为了实现高效的视频直播,选择合适的工具和协议至关重要。 RTMP(Real-Time Messagi…...

从API调用到完整应用:手把手教你用Dashscope和Streamlit搭建一个多模态聊天机器人

从API调用到完整应用&#xff1a;手把手教你用Dashscope和Streamlit搭建多模态聊天机器人 在AI技术快速落地的今天&#xff0c;将强大的API能力转化为直观可用的产品已成为开发者的核心技能。想象一下&#xff0c;你只需要200行Python代码&#xff0c;就能构建一个能"看懂…...

思科ASA防火墙“升级困境“破解“——飞将让50人团队平滑过渡远程办公

一、客户需求介绍 一家50人规模的企业服务公司&#xff0c;此前使用思科ASA 5506防火墙承载本地上网和远程办公需求&#xff0c;但因以下需求陷入瓶颈&#xff1a; 思科ASA 5506​性能不足​&#xff0c;设备自带的AnyConnect许可证不够用&#xff1b;保留移动办公员工习惯&…...

OpenClaw+千问3.5-9B数据清洗:Excel表格异常值检测与修复

OpenClaw千问3.5-9B数据清洗&#xff1a;Excel表格异常值检测与修复 1. 为什么需要AI辅助数据清洗&#xff1f; 上周处理一份客户调研数据时&#xff0c;我遇到了典型的数据清洗难题——表格里混杂着空值、格式混乱的日期、重复记录和错误拼写。手动处理不仅耗时&#xff0c;…...

云原生数据库的设计与实践:从架构到部署

云原生数据库的设计与实践&#xff1a;从架构到部署 前言 作为一个在数据深渊里捞了十几年 Bug 的女码农&#xff0c;我深知云原生技术对数据库的影响。随着云计算的快速发展&#xff0c;云原生数据库已经成为数据库技术的重要发展方向。今天&#xff0c;我就来聊聊云原生数据库…...

W5500 TCP客户端实战 | 02 - 从寄存器配置到数据收发的完整流程解析

1. W5500网络寄存器配置详解 第一次接触W5500芯片时&#xff0c;我被它密密麻麻的寄存器地址搞得头晕眼花。后来发现只要抓住几个核心寄存器&#xff0c;配置起来就像填快递单一样简单。先说说最关键的四个本地网络寄存器&#xff0c;它们相当于设备的"身份证"&#…...

08_微服务划分与团队人数之监控治理与跨团队协作

微服务划分与团队人数之监控治理与跨团队协作 体系内容 可观测性三支柱:指标、日志、链路追踪 治理要素:SLO、Dashboard、告警分级、容量视图、契约审计 Spring Cloud Alibaba 关联:Nacos、Sentinel、Gateway、RocketMQ、Dubbo 与观测平台协同 跨团队机制:接口契约、消息契…...

如何从零搭建Cubli_Mini:开源自平衡机器人完整制作指南

如何从零搭建Cubli_Mini&#xff1a;开源自平衡机器人完整制作指南 【免费下载链接】Cubli_Mini 项目地址: https://gitcode.com/gh_mirrors/cu/Cubli_Mini Cubli_Mini是一款令人惊叹的开源自平衡立方体机器人项目&#xff0c;它通过三个正交安装的飞轮实现姿态控制&am…...

当PLC网口IP丢了怎么办?用Wireshark抓LLDP包,免费找回施耐德M580的地址

工业现场急救指南&#xff1a;用Wireshark找回施耐德M580 PLC的失踪IP地址 那天下午三点&#xff0c;工厂生产线突然停机&#xff0c;监控系统显示PLC通讯中断。当我冲到控制柜前&#xff0c;发现前任工程师留下的文档里&#xff0c;M580的IP地址记录栏赫然写着"见设备标签…...

【FastAPI】 + SQLAlchemy 异步 ORM 实现完整 CRUD 操作

&#x1f680;从零实战&#xff1a;FastAPI SQLAlchemy 异步 ORM 实现完整 CRUD 操作&#xff08;附完整代码&#xff09; 一、为什么要学「FastAPI SQLAlchemy 异步 ORM」&#xff1f; 在现代 Web 服务中&#xff0c;数据库是核心组件。然而&#xff0c;传统同步操作&#x…...

【物联网】基于STM32F429与TMS320F28377的储能变流器控制软件架构设计

目录 一、双处理器架构设计概述 &#xff08;一&#xff09;异构双核系统定位 &#xff08;二&#xff09;硬件资源协同策略 二、STM32F429ZGT6 核心功能开发 &#xff08;一&#xff09;系统管理模块设计 1. 任务调度与状态监控 2. 多源数据融合存储 &#xff08;二&am…...