Excel常用技能分享与探讨(5-宏与VBA简介 VBA之用户窗体-API调用)
liuian 2025-06-13 14:49 6 浏览
之前提到过很多次关于API的调用,比如添加窗体的最大最小化就会用到。但是这一部分的内容设计的内容比较考验专业性,对小白或者浅浅接触编程的人来说不太友好,要用到什么功能的话不如直接问AI或者百度。
以下为关于 VBA 调用 Windows API 做一些详细说明,包含基础概念、语法结构、常见用法及注意事项,我觉得了解原理即可。
1. Windows API 基础
- API 是什么
Windows API(Application Programming Interface)是微软提供的一组函数库,允许应用程序与操作系统交互,实现底层功能(如窗口管理、文件操作、内存操作等)。 - 为何用 VBA 调用 API
VBA 自身功能有限,通过调用 API 可突破限制,实现高级功能(如窗口控制、系统信息获取、注册表操作等)。 - 深入理解 Windows API 与 VBA 的关联
为什么 Windows API 基于 C/C++?
Windows API 是操作系统原生接口,由 C/C++ 实现,因其高效性和直接操作硬件/系统资源的能力。VBA 作为高级语言,无法直接访问底层功能,需通过 DLL(动态链接库)调用 C/C++ 编译的 API 函数。
VBA 与 C/C++ 的桥梁
- 函数声明:VBA 需模拟 C/C++ 的函数原型(参数类型、返回类型、调用约定)。
- 内存管理:VBA 需处理 C/C++ 风格的指针、字符串和结构体。
- 调用约定:Windows API 使用 stdcall 调用约定(而非 VBA 默认的 cdcel),需通过 Declare 隐式声明。
2. VBA 调用 API 的核心步骤
(1) 声明 API 函数
在 VBA 代码模块顶部使用 Declare 语句声明 API 函数,语法如下:
[Public|Private] Declare [PtrSafe] Function|Sub Name Lib "DllName" _
[Alias "AliasName"] ([ArgList]) [As ReturnType]
- 关键参数:
- PtrSafe:64 位 VBA 必选,表示声明兼容 64 位指针。若 Office 为 64 位,需在 Declare 前加 PtrSafe,并调整指针类型(如 LongPtr)
- Lib "DllName":指定 API 所在的 DLL 文件(如 user32.dll, kernel32.dll)。
- Alias "AliasName":处理函数命名冲突或字符集问题(如 MessageBoxA vs MessageBoxW)。
- 函数名冲突:DLL 中存在同名函数。
- 字符集差异:显式调用 A(ANSI)或 W(Unicode)版本。
Declare PtrSafe Function MessageBox Lib "user32" Alias "MessageBoxW" ( ByVal hWnd As LongPtr, ByVal lpText As String, ByVal lpCaption As String, ByVal uType As Long ) As Long
- 参数传递方式
- ByVal:传递值(如数值、字符串指针)。
- ByRef:传递引用(如结构体、输出缓冲区)。
(2) 参数类型映射
VBA 与 C/C++ 数据类型需对应:
C/C++ 类型 | VBA 类型 | 说明 |
int | Long | 32 位整数 |
DWORD | Long | 无符号 32 位整数 |
HANDLE | LongPtr | 句柄(64 位兼容必须) |
LPCSTR | ByVal String | ANSI 字符串指针 |
LPWSTR | ByVal String | Unicode 字符串指针 |
LPVOID | LongPtr | 通用指针 |
BOOL | Long | 布尔值(0=False,其他=True) |
3. 示例:常用 API 调用
(1) 示例 1:弹出消息框
Declare PtrSafe Function MessageBox Lib "user32" Alias "MessageBoxW" (
ByVal hWnd As LongPtr,
ByVal lpText As String,
ByVal lpCaption As String,
ByVal uType As Long
) As Long
Sub ShowMessage()
MessageBox 0, "Hello, API World!", "VBA API Demo", 64 '64=MB_ICONINFORMATION
End Sub
(2) 示例 2:获取窗口标题
Declare PtrSafe Function GetWindowText Lib "user32" Alias "GetWindowTextW" (
ByVal hWnd As LongPtr,
ByVal lpString As String,
ByVal cch As Long
) As Long
Sub GetNotepadTitle()
Dim hWnd As LongPtr
hWnd = FindWindow("Notepad", vbNullString) '需声明 FindWindow API
If hWnd <> 0 Then
Dim title As String * 255
GetWindowText hWnd, title, 255
MsgBox "窗口标题: " & Left(title, InStr(title, vbNullChar) - 1)
End If
End Sub
4. 关键注意事项
(1) 32/64 位兼容性
- LongPtr 代替 Long:
在 64 位 Office 中,所有指针(如窗口句柄 hWnd)必须使用 LongPtr。 - PtrSafe 关键字:
必须添加 PtrSafe 以避免编译错误。
(2) 字符串处理
- ANSI 与 Unicode:
使用 Alias 区分 A (ANSI) 和 W (Unicode) 版本函数。 - 字符串缓冲区:
预先分配足够空间(如 String * 255)并处理 vbNullChar 终止符。
(3) 错误处理
- API 返回错误码:
使用 GetLastError API 获取错误信息:
Declare PtrSafe Function GetLastError Lib "kernel32" () As Long
- VBA 错误捕获:
结合 On Error GoTo 处理异常。
5. 常见问题解决
(1) 编译错误:Implicit declaration
- 原因:未正确声明 PtrSafe 或参数类型错误。
- 解决:检查 Declare 语句是否包含 PtrSafe 和正确的数据类型。
(2) 运行时错误:Invalid calling convention
- 原因:参数类型或数量不匹配。
- 解决:核对 API 函数原型,确保参数顺序和类型正确。
(3) 内存泄漏
- 原因:未正确释放 API 分配的内存(如 GlobalAlloc)。
- 解决:调用对应的释放函数(如 GlobalFree)。
(4)常见错误原因
错误现象 | 可能原因 | 解决方案 |
程序崩溃 | 参数类型错误或缓冲区溢出 | 检查参数类型和缓冲区大小 |
返回错误码 5(拒绝访问) | 权限不足或句柄无效 | 以管理员身份运行或检查句柄来源 |
返回错误码 87 | 参数错误(如结构体未初始化) | 检查参数顺序和结构体定义 |
6. 进阶技巧
- 回调函数:
使用 AddressOf 传递 VBA 函数指针(需模块级声明)。 - 结构体参数:
定义与 C 兼容的 Type 结构体,按引用传递(ByRef)。 - DLL 动态加载:
通过 LoadLibrary 和 GetProcAddress 动态调用 API。
7. 资源推荐
- API 文档:Microsoft Docs
- API 常量定义:Win32API.txt(可从网络获取预定义常量)
- 调试工具:Spy++(查看窗口句柄)、API Monitor(监控 API 调用)
- 查看Dll中函数的方法:直接搜索Depends工具下载安装
通过合理使用 Windows API,VBA 的功能边界可大幅扩展,但需注意稳定性与兼容性。建议先在测试环境中验证代码逻辑。
相关推荐
- 面试怕被问Hashmap,多看看这个文章
-
o数据结构otable数组长度永远为2的幂次方o那么为什么要把数组长度设计为2的幂次方呢?o扩容o链表树化o红黑树拆分o查找o插入o删除o遍历oequasl和hashcode总结HashMap是面试中...
- 非常简洁地重试Retry组件,使用起来杠杠的
-
前言小伙伴是不是经常遇到接口调用异常,超时的场景?尤其网络抖动导致timeout超时的场景,我们一般产品就会叫我们要重试几次。很多小伙伴的实现方式是写个循环调用for(inti=1;i<=3;...
- Kafka消息可靠传输之幂等、事务机制
-
一般而言,消息中间件的消息传输保障有3个层级,分别如下。atmostonce:至多一次。消息可能会丢失,但绝对不会重复传输。atleastonce:最少一次。消息绝不会丢失,但可能会重复传输。...
- Seata源码—9.Seata XA模式的事务处理
-
大纲1.SeataXA分布式事务案例及AT与XA的区别2.SeataXA分布式事务案例的各模块运行流程3.Seata使用SpringBoot自动装配简化复杂配置4.全局事务注解扫描组件的自动装配...
- Disruptor—3.核心源码实现分析一
-
大纲1.Disruptor的生产者源码分析2.Disruptor的消费者源码分析3.Disruptor的WaitStrategy等待策略分析4.Disruptor的高性能原因5.Disruptor高性...
- Spring Boot 进阶-详解SpringBoot中条件注解使用
-
作为使用SpringBoot框架的开发者来讲,如果你连如下的这些注解你都没有听说过,没有用过,那我劝你还是放弃吧?在SpringBoot中我们最常见到的注解应该是条件注解了吧!也就是@Condit...
- 如何自定义编解码器(如何自定义编解码器的程序)
-
1.前言上一节我们一节了解了什么是编码解码、序列化和反序列化了,并且留有一道思考题,本节内容主要是深入解析该思考题。思考题:能否把我们的编码和解码封装成独立的Handler呢?那么应该如何去封装...
- Disruptor—3.核心源码实现分析二
-
大纲1.Disruptor的生产者源码分析2.Disruptor的消费者源码分析3.Disruptor的WaitStrategy等待策略分析4.Disruptor的高性能原因5.Disruptor高性...
- 线程的状态有哪些?它是如何工作的?
-
线程的状态有哪些?它是如何工作的?线程(Thread)是并发编程的基础,也是程序执行的最小单元,它依托进程而存在。一个进程中可以包含多个线程,多线程可以共享一块内存空间和一组系统资源,因此线程之间的切...
- 有图解有案例,我终于把Condition的原理讲透彻了
-
平时加解锁都是直接使用Synchronized关键字来实现的,简单好用,为啥还要引用ReentrantLock呢?为了解决小伙伴的疑问,我们来对两者做个简单的比较吧:相同点两者都是“可重入锁”,即当前...
- 白话DUBBO原理,通俗易记,再也不怕面试时讲不清楚了
-
现在的各种面试免不了要问些中间件,尤其是互联网公司,更注重获选人对中间件的掌握情况。在中间件中,有一大类是关于RPC框架的,Dubbo即是阿里出品的一款很著名的RPC中间件,很多互联网公司都在用,面试...
- Java 最细的集合类总结(java常用的集合类有哪些)
-
数据结构作为每一个开发者不可回避的问题,而Java对于不同的数据结构提供了非常成熟的实现,这一个又一个实现既是面试中的难点,也是工作中必不可少的工具,在此,笔者经历漫长的剖析,将其抽丝剥茧的呈现出...
- 详解Java异常(Exception)处理及常见异常
-
很多事件并非总是按照人们自己设计意愿顺利发展的,经常出现这样那样的异常情况。例如:你计划周末郊游,计划从家里出发→到达目的→游泳→烧烤→回家。但天有不测风云,当你准备烧烤时候突然天降大雨,只能终止郊...
- 为什么阿里强制要求不要在foreach循环里进行元素remove和add操作
-
在阅读《阿里巴巴Java开发手册》时,发现有一条关于在foreach循环里进行元素的remove/add操作的规约,具体内容如下:错误演示我们首先在IDEA中编写一个在foreach循...
- SpringBoot条件化配置(@Conditional)全面解析与实战指南
-
一、条件化配置基础概念1.1什么是条件化配置条件化配置是Spring框架提供的一种基于特定条件来决定是否注册Bean或加载配置的机制。在SpringBoot中,这一机制通过@Conditional...
- 一周热门
-
-
Python实现人事自动打卡,再也不会被批评
-
Psutil + Flask + Pyecharts + Bootstrap 开发动态可视化系统监控
-
【验证码逆向专栏】vaptcha 手势验证码逆向分析
-
一个解决支持HTML/CSS/JS网页转PDF(高质量)的终极解决方案
-
再见Swagger UI 国人开源了一款超好用的 API 文档生成框架,真香
-
网页转成pdf文件的经验分享 网页转成pdf文件的经验分享怎么弄
-
C++ std::vector 简介
-
python使用fitz模块提取pdf中的图片
-
《人人译客》如何规划你的移动电商网站(2)
-
Jupyterhub安装教程 jupyter怎么安装包
-
- 最近发表
- 标签列表
-
- python判断字典是否为空 (50)
- crontab每周一执行 (48)
- aes和des区别 (43)
- bash脚本和shell脚本的区别 (35)
- canvas库 (33)
- dataframe筛选满足条件的行 (35)
- gitlab日志 (33)
- lua xpcall (36)
- blob转json (33)
- python判断是否在列表中 (34)
- python html转pdf (36)
- 安装指定版本npm (37)
- idea搜索jar包内容 (33)
- css鼠标悬停出现隐藏的文字 (34)
- linux nacos启动命令 (33)
- gitlab 日志 (36)
- adb pull (37)
- table.render (33)
- uniapp textarea (33)
- python判断元素在不在列表里 (34)
- python 字典删除元素 (34)
- vscode切换git分支 (35)
- python bytes转16进制 (35)
- grep前后几行 (34)
- hashmap转list (35)