当前位置:首页 > 投稿 > 正文内容

VB/VBA中Variant的坑,却是传参、动态调用API的利器

福瑞号2023-02-04 14:05:10投稿145
VB/VBA中Variant的坑,却是传参、动态调用API的利器-图1

在VB中要有所为,是绕不开Variant滴

前言
在VB/VBA中Variant类型,想必各位都不陌生,因为绝大部分VB/VBA内置函数的参数和返回值,都是这货。很多讨论VB/VBA性能优化的话题,也不会忘记这货。笔者也在《Variant构造智能指针,撒欢了玩》一文中,初提Variant的妙处。要深入VB/VBA,笔者甚至在《VB变量的重构,横看成岭侧成峰》一文中,利用Variant重置VB/VBA的数据类型,是进入VB/VBA指针高阶应用的隐秘之门。
作为数据类型的容器,Variant的结构也在前述文章中给予了介绍。既然作为数据容器,最常用的场景,当然就是传递数据啦。本文,笔者就跟大家分享下Variant在数据传递中中的妙用。
一、患难之中求真情
要想体会Variant的妙,自然离不开不妙的参照物。那首先,让我们看看VB/VBA中有哪些“致命”点吧:
1、数据类型不对,编译不让通过
实话说,笔者现想,还真不好找案例,比如下面这种故意,VB/VBA都能放过,也是无语了。
VB/VBA中Variant的坑,却是传参、动态调用API的利器-图2

看到这,就该知道,确认过眼神了

为了说明数据类型不对,对编译的影响,这里给大家介绍VB/VBA中一个神秘的菜单项。VB6位于“运行”下的“全编译执行”,而VBA位于“调试”下的“编译VBAProject”。如下图所示,说不定很多人,都还没注意过这俩兄弟哦,名字很具有误导性,尤其是VBA的。
VB/VBA中Variant的坑,却是传参、动态调用API的利器-图3

VBA是支持编译的

但是,这哥俩却不是用来生成EXE或DLL的,却可以模拟编译过程,检查编译错误。在这里呢,笔者为了方便说明问题,就用这个功能代替编译检查。于是,又列举了下图所示的例子,然后进行编译检查。
VB/VBA中Variant的坑,却是传参、动态调用API的利器-图4

将String传递给Long类型,会发生什么?

结果很显然,会发生类型不匹配的错误,如下图所示:
VB/VBA中Variant的坑,却是传参、动态调用API的利器-图5

在传参时,类型匹配,是最起码的

2、传数组,动不动就红色告警,尽耽误事儿
数组在任何一种编程语言中,都占据着举足轻重的地位。尤其在业界,普遍认为VB/VBA的性能存在某种问题的情况下,数组就成为很多人提高性能的法宝。相信很多人都曾遇到过,要将数组作为参数进行传递,尽扯拐,比如下面这:
VB/VBA中Variant的坑,却是传参、动态调用API的利器-图6

难道是Optional的锅?

好吧,那删掉吧,默认使用ByRef好了,再来...
VB/VBA中Variant的坑,却是传参、动态调用API的利器-图7

诸位,就没遇见过?就是这么神奇!

什么叫参数必须为变体?那前面的Integer,String等,就叫变体了!什么是缺省的内部类型?数组不是数据类型,不能传递?你是不是默默地将Optional关键字取消了!尽管,有时候这个参数组,一个参数也没有,是不是也会强颜欢笑地,随便传个什么,来表示表示?
3、函数指针,动态调用
关于VB/VBA中的指针问题,前面很多文章里都提到了,读者朋友们可自行关注翻阅以往的文章、头条内容。其实在VB/VBA工程内部,AddressOf关键字就可以获取模块中的函数指针,但是意义不大。因为工程内部,直接可以Call函数名就能使用了,不必大费周章。函数指针,重要的是使用非工程内部的函数,尤其在很多讨巧领域,使用未导出函数。
在VB/VBA中可以使用Declare语句来声明要使用的函数,但是这货不仅效率低,开销大,关键是它是标准调约的呀,而且所用函数必须导出。这意味着什么?
大量优质的C库资源(C调约)没法用了吧,Sqlite数据库不能用了吧,系统很多API不能用了吧,很多讨巧的技能用不上了...少了资源不说,能用的资源又能用好了么?VB/VBA都到指针份上了,谁还会忍受Declare下的各种重复的复制粘贴?VB/VBA号称为自动化服务的,可是连小小的动态调用都搞不定,还谈什么自动化胶水呢?
或许,很多人会觉得函数指针挺简单的,GetModuleHandle+GetProcAddress不就行了。获取函数指针,是很简单。但凡事都有然而,不可能每个函数都重复配套一次吧,所以封装动态调用函数的重要性就出来了。
所以这里的动态调用,其实是指封装1个万能调用函数,对就像Invoke函数那样。相信很多朋友都曾尝试过,笔者也见过很多实现,但都很蹩脚,其中尤其是参数的处理。我们知道,Win32的API成千上万,更不要说第三方库中的API了。这么多函数,参数的个数、类型肯定是不一样的。如何在一个万能调用函数中,处理这些不同呢?
二、患难中见真情
限于篇幅和时间精力,上述列举的只是VB/VBA编程中诸多问题中的冰山一角。若是,未能深入掌握VB/VBA的语言特点,想必很容易开始嫌弃VB/VBA,从而投向其他专业工具。究竟是福是祸,这就不好说了。毕竟人生苦短,能简单何必复杂!
1、传参类型不匹配,Variant可是传参中的Any
遇到『一.1』的问题,在VB/VBA中其实很容易解决,简单地将参数类型声明为Any即可。关于Any类型,在VB/VBA的圈子里,也有很多传说。最信誓旦旦的解释,莫不过将其视为提高效率的某种神秘的危险做法。
VB/VBA中Variant的坑,却是传参、动态调用API的利器-图8

Any,其实不是一种类型

其实,Any不算是一种数据类型,它只是告诉编译器,不检查数据类型而已。被Any修饰的参数,无论传递什么类型过去,编译时都不会报错,实际上却很可能发生内存溢出错误。所以,有人觉得它危险。再有,被Any修饰的参数,不能与ByVal关键字合作,不能传值,而只能传址,所以有人觉得它高效。
笔者在《ByVal与ByRef,究竟谁更高效?》详细地介绍了参数传递过程中,实参与形参的一些特点,未阅读过的读者,可关注阅读。因为编译器不检查数据类型,实参传递给形参时,形参因不知道要分配的内存大小,就无法分配内存。所以,只能传址进行引用,而不能传值。
Any类型,对于VB/VBA使用API而言,只要理解了上述关系,并准确使用,不仅可以提高性能,最核心的是可以极大提高API使用的灵活性。就像上图所示,不需要显式传递字符串指针给lstrlenA函数,仅需要按平常所理解的那样,将字符串变量丢给Any修饰的参数,就可以自动传递字符串指针过去。
不过,Any再好,那也只是Declare的配套,跟自己的函数沾不上边的。不过,还有Variant呀,Variant就是非Declare函数的Any。VB/VBA中的所有数据类型,都可以传递给Variant。
2、Variant传数组,Optional ByVal/ByRef,就不是个事儿
『一.2』中的问题,涉及到数组参数的传递。在VB/VBA中,传递数组变量时,只能ByRef方式,而且不能使用Optional关键字。这无疑对于重要的数组来讲,是不公平的(吐槽点)。其实改用Variant,用这个另类Any,一切就迎刃而解了。
而且,可以ByVal传值,不会影响传入的数组变量。为了形象地说明,笔者写了个Demo测试如下图所示:
VB/VBA中Variant的坑,却是传参、动态调用API的利器-图9

看到没,ByVal传递数组哦

其中VarArrayInfo函数,提取Variant类型数组的数据地址。测试结果表明,数组变量可以ByVal方式传递参数,并不会影响传入的数组。更关键地是,数组可以作为可选参数进行传递了。
3、动态调用函数指针,Variant传参,参数类型和个数不再困扰
『一.3』中的问题,涉及到不定参数的传递问题。在VB/VBA中,可以使用ParamArray关键字修饰参数来实现这一目的。但是参数必须是Variant数组,且不能与Optional关键字组合。
或许有人说,既然Variant可以直接传递数组,还可以与Optional关键字组合,为何不直接使用Variant类型参数呢?这个,怎么说呢,只知其一不知其二啊。
ParamArray修饰的参数,本身可不传递参数啊,这不就是Optional关键字的意思了么。再者,它把数组展开,直接进行传递,多形象直观啊!尤其是,动态调用API,形式上与Declare方式也很像呀,容易理解和检查。
但是,无论是直接使用Variant传递参数数组,还是使用ParamArray关键字传递不定数参数,都必须对Variant有深入地理解。有需要进一步了解Variant机制的朋友,欢迎关注BtOfficer进行咨询哦。
三、Variant不仅是任性的小花,还是惊喜的小花
如果读者朋友们觉得,Variant传递数组,也不过如此的话。那笔者再给出一个测试Demo,看看下图的测试结果:
VB/VBA中Variant的坑,却是传参、动态调用API的利器-图10

Variant虽好,但是坑也挺多的

Variant的确可以传递数组,那当然可以像数组那样来使用数组了,比如上图中的v(0),表示数组中的第一个元素。在VB/VBA中,可以使用VarPtr函数获取非数组变量地址。a(0),不仅表示a数组中首元素地址,更代表首元素变量。所以,VarPtr(a(0))可以获取函数首元素地址。但是,传递给Variant的v(0),却不行了呢?
经验丰富的人员,一看VarPtr(v(0))的返回值,就知道这是栈地址。动态数组数据,都是分配在堆上的。这里面,是有猫腻滴。
其实,这就是Variant另一个让人惊喜的地方,它可以表现得像函数一样。v(0)中,v便是函数名,(0)便是函数列表,返回的元素值,便是函数返回值。还记得笔者以前提过,函数名就是一个临时变量么?如果读者朋友们理解这一点,就很容易理解上面的结果。
怎么样?Variant让你惊喜了吗?Variant传递数组时,可以像数组变量那样获取元素,但却不可以像数组变量那样获取元素地址。
欢迎关注BtOfficer呀(收藏、点赞、关注+转发),更多精彩仍在继续哦(专栏文章更系统,更精彩,但需要支持哦),有严肃的技术,也有轻松的唠嗑,期待你的加入!

扫描二维码推送至手机访问。

版权声明:本文由福瑞号发布,如需转载请注明出处。

本文链接:http://furui.com.cn/60023.html

“VB/VBA中Variant的坑,却是传参、动态调用API的利器” 的相关文章

dnf深渊模式技巧(DNF:提高深渊爆率的几个小技巧)

dnf深渊模式技巧(DNF:提高深渊爆率的几个小技巧)

哈喽,大家好!我是游戏小花酱!你是否还在为深渊爆率低,不出史诗,不出神话而懊恼?今天,小花酱我就告诉大家,能够提高深渊爆率的几个小技巧!赶快收藏起来吧! 一、刷深渊尽量选择人少的频道! 二、在夜晚,凌晨,或早上六点之后刷深渊! 三、电脑越卡越好!最好fps低于20! 四、刷深渊选择普通模式,我觉得...

今日诸城貂皮价格最新行情(2022年1月6日山东诸城水貂皮价格)

今日诸城貂皮价格最新行情(2022年1月6日山东诸城水貂皮价格)

在诸城皮毛市场上,水貂生皮的数量比上个大集要少了,买皮的也少了,成交也不活跃。现在市场上买皮的也多是囤皮,购买意愿不强烈,只有价格合适他们才会购买。 山东诸城水貂皮价格 现在熟板销售基本上是停滞了,买生皮的也是卖了明年待售。虽然快过年了,买皮的少了,但是价格比以前并不低,而买皮的觉得不便宜还不...

昆明到丽江车程(旅游专线列车—K9686)

昆明到丽江车程(旅游专线列车—K9686)

感受云南最高端的旅游专列:昆明—丽江民族文化品牌列车 2014年9月12日上午12:10分,昆明铁路局开行的昆明——丽江民族文化品牌列车启动运行,一直行驶至今。 昆明站21:15分发车,途径大理站,次日7:00到达丽江站,车程7小时34分。 丽江号采用双层客车车厢,外装融入了纳西族、白族、彝族等云南...

maigoo十大品牌网(口红牌子排行榜10强)

maigoo十大品牌网(口红牌子排行榜10强)

摘要:口红是一类唇用美容化妆品,其主要功能是赋予嘴唇以色调,强调或改变两唇的轮廓,显示出更有生气和活力。那么口红品牌有哪些,下面有maigoo小编为你推荐口红十大品牌、口红品牌哪个好,供你参考。 口红是唇用美容化妆品的一种。口红包括唇膏、唇棒、唇彩、唇釉等,它能让唇部红润有光泽,具有滋润、保护嘴唇的...

高达之我主沉浮(这座长三角城市疯狂围填海)

高达之我主沉浮(这座长三角城市疯狂围填海)

◎智谷趋势(ID:zgtrend) | 黄汉城 全中国哪一个城市最充满谜团? 有人说是河西走廊上,那个代号404的城市,它神秘到连名字都没有。也有人说,是两千万人口的帝都,这里有很多地方都是高墙深院,密不透风。 在我眼里,长三角有一个富甲天下的城市同样很迷。 明明这里的民间资本超过一万亿[1],号...

中国邮政编码(我国邮政编码的位数分别代表了什么)

中国邮政编码(我国邮政编码的位数分别代表了什么)

我国邮政编码的位数分别代表了省、邮区和市里的投递局。 1.中国邮政编码实行“四级六码”制,第一位第二位数字表示省,第三位数字表示邮区,第四位数字表示县或者市,最后两位数字表示投递局。例:以324115这 个邮编为例,32表示浙江省(31、32都代表的是浙江省),324表示浙江省衢州市邮政局(三级邮区...