windows C++-windows C++/CX简介(三)
^类型
(^) 是 C++/CX 最突出的功能之一——当人们第一次看到 C++/CX 代码时,很难不注意到它。那么,^ 类型到底是什么?这是类型是一种智能指针类型,它自动管理 Windows 运行时对象的生命周期,也 提供自动类型转换功能以简化 Windows 运行时对象的使用。
我们将首先讨论如何通过 WRL 使用 Windows 运行时对象,然后解释 C++/CX 帽子如何工作以使事情变得更简单。
public interface struct IGetValue{int GetValue() = 0;};public interface struct ISetValue{void SetValue(int value) = 0;};public ref class Number sealed : public IGetValue, ISetValue{public:Number() : _value(0) { }virtual int GetValue() { return _value; }virtual void SetValue(int value) { _value = value; }private:int _value;};
在这个修改后的 Number 实现中,我们定义了一对接口,IGetValue 和 ISetValue,它们声明了 Number 的两个成员函数;然后 Number 实现了这两个接口。除此之外,一切看起来应该非常熟悉。
请注意,Number 实际上实现了三个 Windows 运行时接口:除了 IGetValue 和 ISetValue 之外,编译器仍会生成 Number 实现的 __INumberPublicNonVirtuals 接口。由于 Number 的所有成员都是由显式实现的接口(IGetValue 和 ISetValue)声明的,因此编译器生成的 __INumberPublicNonVirtuals 不会声明任何成员。但是,此接口仍然是必需的,因为它是 Number 类型的默认接口。每个运行时类型都必须有一个默认接口,并且默认接口几乎始终对类是唯一的。稍后我们将看到默认接口为何如此重要。
生命周期管理
Windows 运行时引用类型使用引用计数进行对象生命周期管理。所有 Windows 运行时接口(包括 Number 实现的所有三个接口)都直接派生自 IInspectable 接口,而该接口本身又派生自 COM IUnknown 接口。IUnknown 声明了三个成员函数,用于控制对象的生命周期并允许类型转换。
MSDN对 IUnknown 生命周期管理的工作原理进行了全面概述。但原理非常简单:每当您创建对对象的新引用时,都必须调用 IUnknown::AddRef 来增加其引用计数;每当您“销毁”对对象的引用时,都必须调用 IUnknown::Release 来减少引用计数。引用计数初始化为零,在对 AddRef 和 Release 进行一系列调用后,当引用计数再次达到零时,对象将自行销毁。
当然,在使用 C++ 编程时,我们很少(实际上从不)直接调用 AddRef 和 Release。相反,我们应该尽可能地使用智能指针,在需要时自动进行这些调用。使用智能指针有助于确保对象不会因错过 Release 而泄漏,也不会因过早 Release 或 AddRef 失败而过早销毁。
ATL 包括 CComPtr 和一系列相关的智能指针,它们长期以来一直用于 COM 编程,用于自动管理实现 IUnknown 的对象的引用计数。WRL 包括 ComPtr,它是一种改进和现代化的 CComPtr(改进示例:ComPtr 不会像 CComPtr 那样重载一元 &)。
对于那些没有做过太多 COM 编程并且不熟悉 ComPtrs 的人:如果您使用过 shared_ptr(包含在 C++11、C++ TR1 和 Boost 中),ComPtr 在生命周期管理方面实际上具有相同的行为。机制不同(ComPtr 使用 IUnknown 提供的内部引用计数,而 shared_ptr 支持任意类型,因此必须使用外部引用计数),但生命周期管理行为相同。
C++/CX hat 具有与 ComPtr 完全相同的生命周期管理语义。复制 T^ 时,会调用 AddRef 来增加引用计数,而当 T^ 超出范围或被重新分配时,会调用 Release 来减少引用计数。我们可以考虑一个简单的示例来演示引用计数行为:
{T^ t0 = ref new A();T^ t1 = ref new B();t0 = t1;t0 = nullptr;}
首先,我们创建一个 A 对象并将其所有权赋予 t0。此 A 对象的引用计数为 1,因为有一个 T^ 引用了它。然后,我们创建一个 B 对象并将其所有权赋予 t1。此 B 对象的引用计数也是 1。
t0 = t1 的最终结果是 t0 和 t1 都指向同一个对象。这必须分三步完成。首先,调用 t1->AddRef() 来增加 B 对象的引用计数,因为 t1 正在获得该对象的所有权。其次,调用 t0->Release() 来释放 t0 对 A 对象的所有权。这会导致 A 对象的引用计数降至零,并且 A 对象会自行销毁。这会导致 B 对象的引用计数增加到 2。第三,也是最后,将 t1 设置为指向 B 对象。
然后,我们分配 t0 = nullptr。这会将 t0 “重置”为空,从而导致其释放对 B 对象的所有权。这会调用 t0->Release(),导致 B 对象的引用计数减少到 1。
最后,执行将到达块的结束括号:}。此时,所有局部变量都以相反的顺序被销毁。首先,t1 被销毁(智能指针,而不是指向的对象)。这会调用 t1->Release(),导致 B 对象的引用计数降至零,因此 B 对象会自行销毁。然后销毁 t0,这是一个无操作,因为它为空。
如果我们只关心生命周期管理,那么实际上根本不需要 ^:ComPtr<T> 足以管理对象生命周期。
类型转换
在 C++ 中,涉及类类型的某些类型转换是隐式的;其他类型转换可以使用强制转换或一系列强制转换来执行。例如,如果 Number 及其实现的接口是普通的 C++ 类型而不是 Windows 运行时类型,则从 Number* 到 IGetValue* 的转换将是隐式的,我们可以使用 static_cast 或 dynamic_cast 从 IGetValue* 转换为 Number*。
这些转换不适用于 Windows 运行时引用类型,因为引用类型的实现是不透明的,并且引用类型在内存中的布局未指定。在 C# 中实现的引用类型在内存中的布局可能与在 C++ 中实现的等效引用类型不同。因此,在直接使用 Windows 运行时类型时,我们不能依赖 C++ 语言特定的功能,例如隐式派生到基转换和强制转换。
要执行这些转换,我们必须改用 IUnknown 接口的第三个成员函数:IUnknown::QueryInterface。此成员函数可视为与语言无关的 dynamic_cast:它尝试执行到指定接口的转换并返回转换是否成功。由于每个运行时类型都实现 IUnknown 接口并为 QueryInterface 提供自己的定义,因此它可以执行任何必要的操作,以在实现它的语言和框架中获取正确的接口指针。
相关文章:
windows C++-windows C++/CX简介(三)
^类型 (^) 是 C/CX 最突出的功能之一——当人们第一次看到 C/CX 代码时,很难不注意到它。那么,^ 类型到底是什么?这是类型是一种智能指针类型,它自动管理 Windows 运行时对象的生命周期,也 提供自动类型转换功能以简化…...
《黑神话.悟空》:一场跨越神话与现实的深度探索
《黑神话.悟空》:一场跨越神话与现实的深度探索 在国产游戏日益崛起的今天,《黑神话.悟空》以其独特的剧情、丰富的人物设定和深刻的主题,成为了无数玩家翘首以盼的国产3A大作。这款游戏不仅是一次对传统故事的创新演绎,更是一场对…...
【Kotlin设计模式】建造者模式在Android中的应用
前言 建造者模式(Builder Pattern)是一种创建型设计模式,一步一步地构建一个复杂对象的不同部分,而不是直接创建该对象的实例。建造者模式的核心思想是将对象的构建过程与其表示分离,使得同样的构建过程可以创建不同的…...
Kafka 性能为什么比 RocketMQ 好
Kafka 性能更好的原因 因为 kafka 零拷贝技术跟 RocketMQ 的不一样。 kafka 零拷贝技术使用的是 sendfileDMA scatter/gather 。只需要经过 2 次拷贝,2 次上下文切换RocketMQ 零拷贝使用的 mmap 内存映射,需要经过 3 次拷贝,4 次上下文切换…...
el-image的配套使用(表格,表单)
1. 配合table在一起使用,支持预览 此处使用场景是表格中只显示一张图片 preview-src-list只支持数组,故需要将单个字符串转换为转换为字符串数组 <el-table-column align"center" label"二维码"><template slot-scope&q…...
MKS MWH-5匹配器Automatc matching impedance Network手侧
MKS MWH-5匹配器Automatc matching impedance Network手侧...
打卡50天------图论
正式开启图论了,作为一个前端工程师,这个代码随想录真的刷新了我对于算法的认知,每天都在学习新东西。 别着急、放轻松、慢慢来。 一、图论理论基础 二、深搜理论基础 了解一下深搜的原理和过程,其实对于深搜和广搜我自己也写过…...
实现 FastCGI
CGI的由来: 最早的 Web 服务器只能简单地响应浏览器发来的 HTTP 请求,并将存储在服务器上的 HTML 文件返回给浏 览器,也就是静态 html 文件,但是后期随着网站功能增多网站开发也越来越复杂,以至于出现动态技 术&…...
0x01 GlassFish 任意文件读取漏洞复现
参考文章: 应用服务器glassfish任意文件读取漏洞 - SecPulse.COM | 安全脉搏 fofa 搜索使用该服务器的网站 网络空间测绘,网络空间安全搜索引擎,网络空间搜索引擎,安全态势感知 - FOFA网络空间测绘系统 "glassfish"&…...
RLOC_ORIGIN
RLOC_ORIGIN属性为相对放置的对象提供绝对位置或LOC RTL设计中的宏(RPM)。有关定义RPM和使用 RLOC_ORIGIN属性,请参阅《Vivado Design Suite用户指南:使用约束》 (UG903)[参考文献19]。 RPM是通过使用H_set…...
【Python】成功解决 NameError: name ‘reload‘ is not defined
【Python】成功解决 NameError: name ‘reload’ is not defined 下滑即可查看博客内容 🌈 欢迎莅临我的个人主页 👈这里是我静心耕耘深度学习领域、真诚分享知识与智慧的小天地!🎇 🎓 博主简介:985高校…...
Android.bp和Android.mk文件有的区别
文章目录 1. 构建系统2. 语法和格式3. 可维护性和扩展性4. 编译效率5. 未来趋势 在Android的构建系统中, Android.mk和 android.bp是用于定义如何编译项目文件的两种文件类型,它们有一些显著的区别。 1. 构建系统 Android.mk:使用于基于GN…...
思科设备静态路由实验
拓扑及需求 网络拓扑及 IP 编址如图所示;PC1 及 PC2 使用路由器模拟;在 R1、R2、R3 上配置静态路由,保证全网可达;在 R1、R3 上删掉上一步配置的静态路由,改用默认路由,仍然要求全网可达。 各设备具体配置…...
学习笔记第二十九天
IPC 进程间通信方式:共享内存 原理 共享内存是最高效的进程间通信方式之一,因为它允许两个或多个进程直接访问同一块物理内存区域。这种机制避免了数据在用户空间和内核空间之间的频繁拷贝,从而显著提高了数据传输的效率。 在Linux系统中&…...
Apache Paimon走在正确的道路上|一些使用体验和未来判断
Apache Paimon这个框架大家应该都不陌生了。 在实际工作中大家应该多多少少都用到,这个文章是一个简单的使用体会。不涉及湖框架的拉踩,我们的着眼点是解决实际问题。 我来结合自身体会跟大家说说Paimon这个框架和对未来的一些判断。大家可以参考&#x…...
安装MySQL入门基础指令
一.安装MySQL(以5.7版本为例) 1.一路默认安装,截图供大家参考 修改自己window安装名字即可 2.配置环境变量 C:\Program Files\MySQL\MySQL Server 5.7\bin 写入系统环境变量即可在window窗口使用其服务了 3.登录MySQL服务 进入控制台输入命令 mysql -u root …...
搜维尔科技:【研究】Haption Virtuose外科手术触觉视觉学习系统的开发和评估
Haption面临挑战 除此之外,外科医生有时会对骨组织进行非常复杂的手术,其中一个例子是人工耳蜗的手术植入。重要的是要避免神经或血管等危险结构受伤,并尽可能轻柔地进行手术。在外科医生能够安全、无差错地进行此类手术之前,需要…...
达梦表字段、字段类型,精度比对及更改字段SQL生成
达梦表字段、字段类型,精度比对及更改字段SQL生成: 依赖 <!-- 达梦 Connector --><dependency><groupId>com.dameng</groupId><artifactId>DmJdbcDriver18</artifactId><version>8.1.3.62</version>&l…...
2.pandas--读取文件夹中所有excel文件进行合并
文章目录 代码对应的本地文件文件夹目录三个文件夹中的内容test01.xlsxtest02.xlsxtest03.xlsx 三个文件合并后得到merge.xlsx文件文件内容 生成result.xlsx文件内容 代码 import glob import pandas as pddf_merge pd.DataFrame() # 创建一个空的DataFramefolder_path &qu…...
WPS Office两个严重漏洞曝光,已被武器化且在野利用
WPS Office作为一款用户基数超过2亿的广泛使用的办公套件,被发现存在两个关键漏洞(CVE-2024-7262和CVE-2024-7263),这些漏洞可能导致用户遭受远程代码执行攻击。这两个漏洞的CVSS评分为9.3,表明它们的严重性很高&#…...
XCTF-web-easyupload
试了试php,php7,pht,phtml等,都没有用 尝试.user.ini 抓包修改将.user.ini修改为jpg图片 在上传一个123.jpg 用蚁剑连接,得到flag...
应用升级/灾备测试时使用guarantee 闪回点迅速回退
1.场景 应用要升级,当升级失败时,数据库回退到升级前. 要测试系统,测试完成后,数据库要回退到测试前。 相对于RMAN恢复需要很长时间, 数据库闪回只需要几分钟。 2.技术实现 数据库设置 2个db_recovery参数 创建guarantee闪回点,不需要开启数据库闪回。…...
阿里云ACP云计算备考笔记 (5)——弹性伸缩
目录 第一章 概述 第二章 弹性伸缩简介 1、弹性伸缩 2、垂直伸缩 3、优势 4、应用场景 ① 无规律的业务量波动 ② 有规律的业务量波动 ③ 无明显业务量波动 ④ 混合型业务 ⑤ 消息通知 ⑥ 生命周期挂钩 ⑦ 自定义方式 ⑧ 滚的升级 5、使用限制 第三章 主要定义 …...
2025 后端自学UNIAPP【项目实战:旅游项目】6、我的收藏页面
代码框架视图 1、先添加一个获取收藏景点的列表请求 【在文件my_api.js文件中添加】 // 引入公共的请求封装 import http from ./my_http.js// 登录接口(适配服务端返回 Token) export const login async (code, avatar) > {const res await http…...
VTK如何让部分单位不可见
最近遇到一个需求,需要让一个vtkDataSet中的部分单元不可见,查阅了一些资料大概有以下几种方式 1.通过颜色映射表来进行,是最正规的做法 vtkNew<vtkLookupTable> lut; //值为0不显示,主要是最后一个参数,透明度…...
Android Bitmap治理全解析:从加载优化到泄漏防控的全生命周期管理
引言 Bitmap(位图)是Android应用内存占用的“头号杀手”。一张1080P(1920x1080)的图片以ARGB_8888格式加载时,内存占用高达8MB(192010804字节)。据统计,超过60%的应用OOM崩溃与Bitm…...
pikachu靶场通关笔记22-1 SQL注入05-1-insert注入(报错法)
目录 一、SQL注入 二、insert注入 三、报错型注入 四、updatexml函数 五、源码审计 六、insert渗透实战 1、渗透准备 2、获取数据库名database 3、获取表名table 4、获取列名column 5、获取字段 本系列为通过《pikachu靶场通关笔记》的SQL注入关卡(共10关࿰…...
#Uniapp篇:chrome调试unapp适配
chrome调试设备----使用Android模拟机开发调试移动端页面 Chrome://inspect/#devices MuMu模拟器Edge浏览器:Android原生APP嵌入的H5页面元素定位 chrome://inspect/#devices uniapp单位适配 根路径下 postcss.config.js 需要装这些插件 “postcss”: “^8.5.…...
HarmonyOS运动开发:如何用mpchart绘制运动配速图表
##鸿蒙核心技术##运动开发##Sensor Service Kit(传感器服务)# 前言 在运动类应用中,运动数据的可视化是提升用户体验的重要环节。通过直观的图表展示运动过程中的关键数据,如配速、距离、卡路里消耗等,用户可以更清晰…...
Python基于历史模拟方法实现投资组合风险管理的VaR与ES模型项目实战
说明:这是一个机器学习实战项目(附带数据代码文档),如需数据代码文档可以直接到文章最后关注获取。 1.项目背景 在金融市场日益复杂和波动加剧的背景下,风险管理成为金融机构和个人投资者关注的核心议题之一。VaR&…...
