目录
- [一. 简介](#一. 简介)
- [二. 程序代码](#二. 程序代码)
-
- [2.1 管理员权限执行脚本](#2.1 管理员权限执行脚本)
- [2.2 自定义函数](#2.2 自定义函数)
- [2.3 注册代码](#2.3 注册代码)
- [三. 注册成功](#三. 注册成功)
一. 简介
在C# 创建vba用的类库的文章中,介绍了创建vba类库的方式。
但是vba无法直接使用.dll文件,必须将.dll文件注册到系统中,然后vba才能调用。
vba依赖于Excel,而Excel又有32位和64位这两种版本
- 32位Excel:使用
$env:WINDIR\Microsoft.NET\Framework\v4.0.*\RegAsm.exe注册 - 64位Excel:使用
$env:WINDIR\Microsoft.NET\Framework64\v4.0.*\RegAsm.exe注册
RegAsm.exe是系统自带的,一般在上面的路径中都会存在,为了脚本的通用性,在注册的时候,会同时注册32位和64位。保证所有位数的Excel都可以使用。
🔷当使用RegAsm.exe将.dll类库注册成功之后,会将.dll类库的相关信息写入注册表,效果如下图所示:
$ProgId = "ToolLib.ToolCom":C# 创建vba用的类库文章中提到的ProgId{6143FB0B-9C17-4859-860C-6DA4A466ECD1}:.dll类库关联的Guid

二. 程序代码
2.1 管理员权限执行脚本
🔷使用RegAsm.exe时,必须通过管理员权限,因此在脚本中加入管理员权限判断。若不是管理员,则转换为管理员权限后再执行脚本。
powershell
param(
[string]$originalScriptFolderPath
)
# 校验当前用户是否有管理员权限
function Test-IsAdmin {
$id = [System.Security.Principal.WindowsIdentity]::GetCurrent()
$principal = New-Object System.Security.Principal.WindowsPrincipal($id)
if ($principal.IsInRole([System.Security.Principal.WindowsBuiltInRole]::Administrator)) {
return $true
}
return $false
}
# 若没有管理员权限
if (-not (Test-IsAdmin)) {
Write-Host "未检测到管理员权限,正在尝试重新启动为管理员模式..." -ForegroundColor Yellow
# 暂停1秒钟, 显示文字
Start-Sleep -Seconds 1
# 以管理员方式启动当前脚本
Start-Process -FilePath "powershell.exe" `
-ArgumentList "-NoProfile -ExecutionPolicy RemoteSigned -File $($PSCommandPath) -originalScriptFolderPath $($PSScriptRoot)" `
-Verb RunAs
# 退出当前脚本, 防止打开多个窗口
exit
}
Write-Host "已成功以管理员权限运行PowerShell ...`n" -ForegroundColor Green
# 当以管理员权限运行脚本的时候, 作业目录默认会移动到【C:\WINDOWS\system32】下
# 我们手动将作业目录其移动到脚本所在的文件夹路径
if ($null -ne $originalScriptFolderPath) {
Set-Location $originalScriptFolderPath
}
2.2 自定义函数
🔷注册Com所需的参数
/codebase- 告诉COM:DLL文件所在的绝对路径
- 会在注册表的指定项的InprocServer32中写入.dll文件的绝对路径
/tlb- 给COM客户端(尤其 VBA)用的接口描述文件
- 里面描述了接口, 方法, 参数类型
- 非必须参数
- 但是添加了这个参数之后, vba可以在【工具】→【引用】中找到自定义的
.dll - 然后就可以进行早绑定了
- 但是添加了这个参数之后, vba可以在【工具】→【引用】中找到自定义的
powershell
function Register-Com {
param (
[string]$dllFullPath,
[string]$regAsmExeFullPath
)
$versionMsg = "32位"
if ($regAsmExeFullPath.Contains('Framework64')) {
$versionMsg = "64位"
}
$params = "`"$dllFullPath`" /codebase /tlb"
$result = $null
try {
$result = Start-Process $regAsmExeFullPath -ArgumentList $params -Wait -PassThru
if ($result.ExitCode -eq 0) {
Write-Host "${versionMsg}的dll注册成功"
}
}
catch {
Write-Host "${versionMsg}的dll注册失败: $($result.ExitCode)"
Write-Host "${versionMsg}的dll注册失败: $($_.Exception.Message)"
}
}
powershell
# 卸载Com
function Unregister-Com {
param (
[string]$dllFullPath,
[string]$regAsmExeFullPath
)
$versionMsg = "32位"
if ($regAsmExeFullPath.Contains('Framework64')) {
$versionMsg = "64位"
}
# 卸载Com所需的参数
$params = "`"$dllFullPath`" /unregister"
$result = $null
try {
$result = Start-Process $regAsmExeFullPath -ArgumentList $params -Wait -PassThru
if ($result.ExitCode -eq 0) {
Write-Host "${versionMsg}的dll卸载成功"
}
}
catch {
Write-Host "${versionMsg}的dll卸载失败: $($result.ExitCode)"
Write-Host "${versionMsg}的dll卸载失败: $($_.Exception.Message)"
}
}
- 32位的PowerShell, 只能验证32位的com,现在大多数电脑都是64位的
- 因此使用32位的PowerShell时,必须指定路径
powershell
function Test-32ComBoth {
param (
[string]$progId
)
# 32位Powershell的路径
$powershell32 = "$env:WINDIR\SysWOW64\WindowsPowerShell\v1.0\powershell.exe"
$script = @"
try {
New-Object -ComObject '$progId' | Out-Null
Write-Host '${progId}的32位COM存在'
} catch {
Write-Host '${progId}的32位COM不存在'
}
"@
# 测试32位com
& $powershell32 -NoProfile -Command $script
}
- 64位的PowerShell, 只能验证64位的com
- 现在大多数电脑都是64位的,默认的PowerShell也都是64位的,所以不需要指定PowerShell路径
powershell
function Test-64ComBoth {
param (
[string]$progId
)
# 测试64位com
try {
New-Object -ComObject $progId | Out-Null
Write-Host "${progId}的64位COM存在"
}
catch {
Write-Host "${progId}的64位COM不存在"
}
}
2.3 注册代码
powershell
# 要安装的.dll库的路径, 此处设定为和脚本同一级别的目录下
[string]$dllPath = ".\03-ToolLib.dll"
# .dll库的ProgId
$ProgId = "ToolLib.ToolCom"
# 获取32位和64位的RegAsm.exe的路径, 由于不知道具体版本, 所以使用了通配符
$regAsm32ExePath = "$env:WINDIR\Microsoft.NET\Framework\v4.0.*\RegAsm.exe"
$regAsm64ExePath = "$env:WINDIR\Microsoft.NET\Framework64\v4.0.*\RegAsm.exe"
# 获取RegAsm.exe的绝对路径
$regAsm32ExeFullPath = (Get-ChildItem "$regAsm32ExePath" -ErrorAction SilentlyContinue | Select-Object -First 1).FullName
$regAsm64ExeFullPath = (Get-ChildItem "$regAsm64ExePath" -ErrorAction SilentlyContinue | Select-Object -First 1).FullName
# 如果32位和64位的RegAsm.exe有任意一个不存在的话
if ((-not $regAsm32ExeFullPath) -or (-not $regAsm64ExeFullPath)) {
Write-Host "找不到RegAsm.exe文件..."
Pause
exit 1
}
# 通过相对路径获取绝对路径
$fullPathObj = Resolve-Path $dllPath -ErrorAction SilentlyContinue
if (-not $fullPathObj) {
Write-Host "找不到${dllPath}文件..."
Pause
exit 1
}
# 获取dll文件的绝对路径
$dllFullPath = $fullPathObj.Path
# 让用户选择要执行的操作
[string]$confirm = Read-Host "选择要执行的操作. 1. 安装依赖库 2. 卸载依赖库"
if ($confirm -eq "2") {
# 卸载32位的com
Unregister-Com -dllFullPath $dllFullPath -regAsmExeFullPath $regAsm32ExePath
# 卸载64位的com
Unregister-Com -dllFullPath $dllFullPath -regAsmExeFullPath $regAsm64ExePath
Pause
exit
} elseif ($confirm -ne "1") {
Write-Host "请输入 1 或者 2"
Pause
exit
}
# 注册32位的com
Register-Com -dllFullPath $dllFullPath -regAsmExeFullPath $regAsm32ExePath
# 注册64位的com
Register-Com -dllFullPath $dllFullPath -regAsmExeFullPath $regAsm64ExePath
# .dll类库中的命名空间和方法名
Test-32ComBoth "$ProgId"
Test-64ComBoth "$ProgId"
# 根据 ProgId 查询注册表路径是否存在
$RegistryProgIdPath = "Registry::HKEY_CLASSES_ROOT\$ProgId\"
if (-not (Test-Path $RegistryProgIdPath)) {
Write-Host "找不到${RegistryProgIdPath}注册表..."
Pause
exit 1
}
# 获取 .dll库的Guid
# {6143FB0B-9C17-4859-860C-6DA4A466ECD1}
$dllCLSID = (Get-Item "$RegistryProgIdPath\CLSID").GetValue("")
# 查询32位.dll对应的注册表
Get-ChildItem "Registry::HKLM\Software\Classes\WOW6432Node\CLSID\$dllCLSID" -ErrorAction SilentlyContinue
# 查询64位.dll对应的注册表
Get-ChildItem "Registry::HKEY_CLASSES_ROOT\CLSID\$dllCLSID" -ErrorAction SilentlyContinue
Pause
三. 注册成功
