vs2019 Com组件初探-简单的COM编写以及实现跨语言调用

上一篇实现了如何编写基于IDipatch接口的COM以及vbs如何调用编写的COM

本次主要是实现VBS的CreateObject函数的逻辑

前提条件

  1、掌握C++基础语法

  2、平台安装 vs2019

  3、本地平台为 windows 10 1909 X64

  4、基本的DLL编程知识 (不是必备)

本次目标

  1、创建DLL并实现CreateObject函数

  2、写一个调用DLL的demo

1、创建DLL并实现CreateObject函数

  

  首先通过VS创建一个 动态链接库

  

  

  在编写之前先梳理程序的执行流程

    初始化 Com库

    获取函数指针

    传入参数 

    调用函数指针

    卸载Com库

  

  接下来就开始写我们的DLL

    vs2019 创建DLL项目后系统会默认多出来头文件

      

    以及源文件

      

  我们打开pch.h头文件定义我们的函数声明

    

    参数为 COM组件progID,函数名,参数数量,变长参数

    extern "C" 以C的方式定义

    _declspec(dllimport) 定义此函数为要导出的函数

    

  新建一个ComInit.h 定义Com库的初始化和卸载库函数

 1 // ComInit.h
2
3 #pragma once
4 static bool _init = false;
5
6 // 初始化
7 bool Init();
8
9 // 结束初始化
10 void Release();

  新建一个ComInit.cpp 实现Init和Release函数

// ComInit.cpp

#include "pch.h"
#include "ComInit.h" bool Init()
{
if (_init == true)
{
return _init;
}
else
{
if (S_OK == CoInitialize(NULL))
_init = true;
else
_init = false; return _init;
} return false;
} void Release()
{
if (true == _init)
{
CoUninitialize();
_init = false;
}
}

  之后打开pch.cpp实现CreateObject函数

 1 #include "pch.h"
2 #include "ComStart.h"
3 #include <assert.h>
4 #include <atlbase.h>
5
6 // 报错宏
7 #define ASSERT(s) if((s) == true)
8
9 // Com类名,函数名,传入的参数数量,变长参数
10 VARIANT CreateObject(const WCHAR* __comname,const WCHAR* __funcname,int __count, ...)
11 {
12 /* Com注册到系统后使用 */
13
14 // 是否成功初始化
15 if (true == Init())
16 {
17 // ProgId值存放
18 CLSID clsid;
19
20 // 通过 ProgID 取得组件的 CLSID
21 // CLSID 值存放在注册表 HKEY_CLASSES_ROOT [以__comname加.1为键值(MyCom.FirstClass.1)]
22 HRESULT hr = ::CLSIDFromProgID(__comname, &clsid);
23
24 ASSERT(S_OK != hr)
25 assert(hr != S_OK);
26
27 // 智能指针获取 IUnknow
28 CComPtr<IUnknown>spUnk;
29
30 /*
31 * CoCreateInstance
32 * CLSIDFromProgId获取的值
33 * 指向接口IUnknown的指针
34 * 运行可执行代码的上下文[CLSCTX_ALL 为所有]
35 * IID_IUnknown为返回类型
36 * 用来接收指向Com对象接口地址的指针变量
37 */
38 // 获取IUnknow内容
39 hr = ::CoCreateInstance(clsid, NULL, CLSCTX_ALL, IID_IUnknown, (LPVOID*)&spUnk);
40
41 ASSERT(S_OK != hr)
42 assert(hr != S_OK);
43
44 // 通过IUnknown智能指针,声明新的IDispatch智能指针
45 CComDispatchDriver spDisp(spUnk);
46
47 // 参数数组
48 VARIANT* __args = new VARIANT[__count];
49
50 // 变长参数变量
51 va_list ap;
52
53 // 定位到第一个函数变长参数
54 va_start(ap, __count);
55
56 // 循环获取变长参数,并转换为 VARIANT 类型放入 __args变量
57 for (auto i = 0; i < __count; i++)
58 __args[i] = va_arg(ap, VARIANT);
59
60 // 结束变长参数
61 va_end(ap);
62
63 // Com函数返回值存放
64 VARIANT __ret;
65
66 // 执行Com函数
67
68 /*
69 * [InvokeN]
70 * 函数名
71 * 函数参数
72 * 函数数量
73 * 返回值存放处
74 */
75 hr = spDisp.InvokeN((LPCOLESTR)__funcname, __args, __count, &__ret);
76
77 ASSERT(S_OK != hr)
78 assert(hr != S_OK);
79
80 // 内存回收
81 delete[] __args;
82
83 // 卸载 Com库
84 Release();
85
86 // 返回值
87 return __ret;
88 }
89
90 assert(_init == false);
91 }

   

  完成后编译(CTRL+B)获取到新的dll和lib文件(x64)以及项目的pch.h头文件

    

    

2、写一个调用DLL的demo

  vs2019 新建基于 控制台程序 的项目

  

  移动dll和lib以及pch.h文件到新建项目目录下,并对pch.h文件添加代码

// pch.h

#pragma once
#include <combaseapi.h> // 新添加的代码
#pragma comment(lib,"ComPack.lib") extern "C" _declspec(dllimport) VARIANT CreateObject(const WCHAR * __comname, const WCHAR * __funcname, int __count, ...);

   找到main函数 写入调用代码

#include <iostream>
#include "pch.h" int main()
{
// 参数类型必须为VARIANT
VARIANT __param1; // 参数类型为 LONG
__param1.vt = VT_I4; // 参数值为 2
__param1.lVal = 2; // 获取ComTest.Temp并调用Number 函数 参数数量为1 对Add函数传入参数__param1
VARIANT __ret = CreateObject(L"ComTest.Temp", L"Number",1, __param1); std::cout << __ret.lVal << std::endl;
}

  执行并运行显示执行结果

  

  运行出现错误,检查调用的Com是否已经注册

  如何注册我在上一篇里面有讲过

  

  接下来修改代码尝试调用Wscript.shell里面的Run函数

#include <iostream>
#include "pch.h" int main()
{
VARIANT __param1; // 参数类型为BSTR
__param1.vt = VT_BSTR; // 分配BSTR
__param1.bstrVal = SysAllocString(L"notepad.exe"); // 调用函数并释放BSTR
VARIANT __ret = CreateObject(L"Wscript.shell", L"run",1, __param1);
SysFreeString(__param1.bstrVal);
}

  值得一提的是 COM组件的字符串和以往的字符串有所不同,创建方式和销毁方式也不同

  SysAllocString为创建BSTR字符串

  SysFreeString 为释放BSTR字符串

  运行结果可以看到已经成功的执行了系统命令,打开了一个记事本

  

注意事项:

  com基于IDispatch 接口才可以调用

  Com必须已经注册到系统 (小心误删或者移动路径)

  卸载DLL函数为  regsvr32.exe -ui [DLL未知]

  DLL对应版本尽量一致

github源码:

  3065190005/ComTest: ComTest Code (github.com)

vs2019 Com组件初探-通过IDispatch接口调用Com的更多相关文章

  1. vs2019 Com组件初探-简单的COM编写以及实现跨语言调用

    前提条件 1.掌握C++基础语法 2.平台安装 vs2019 3.本地平台为 windows 10 1909 X64 4.了解vbs基础语法 本次目标 1.掌握Com组件的概念及原理 2.编写一个简单 ...

  2. 【转载】COM 组件设计与应用(十)——IDispatch 接口 for VC.NET

    原文:http://vckbase.com/index.php/wv/1225.html 一.前言 终于写到了第十回,我也一直期盼着写这回的内容耶,为啥呢?因为自动化(automation)是非常常用 ...

  3. 【转载】COM 组件设计与应用(九)——IDispatch 接口 for VC6.0

    原文: http://vckbase.com/index.php/wv/1224.html 一.前言 终于写到了第九回,我也一直期盼着写这回的内容耶,为啥呢?因为自动化(automation)是非常常 ...

  4. Python的扩展接口[2] -> 动态链接库DLL[1] -> 组件对象模型 COM 的 Python 调用

    组件对象模型 COM 的 Python 调用 关于COM的基本概念,可参考组件对象模型 COM的内容,下面主要介绍两种使用 Python 调用 COM 组件的方法. 1 使用 win32com 1.1 ...

  5. wpf 错误 执行了 QueryInterface 调用,请求提供 COM 可见的托管类“BoilerMonitoringV1._0.MapControl”的默认 IDispatch 接口。

    在做wpf嵌入地图时,在自定义的WebBrowser 里面使用JavaScript调用外部方法的时报的错误 在原来的WinForm里 我们只要在窗体类设置的头部设置个 [System.Runtime. ...

  6. COM 组件基础——GUID 和 接口

    一.前言 书接上回,话说在 doc(Word) 复合文件中,已经解决了保存 xls(Excel) 数据的问题了.那么,接下来又要解决另一个问题:当 WORD 程序读取复合文件,遇到了 xls 数据的时 ...

  7. IDispatch接口 - GetIDsOfNames和Invoke(转)

    IDispatch接口是COM自动化的核心.其实,IDispatch这个接口本身也很简单,只有4个方法: IDispatch : public IUnknown { public: virtual H ...

  8. IDispatch接口介绍

    1.         C程序调用时,调用者必须预先知道接口规范(如,参数类型.参数字节长度.参数顺序等).由于不同语言这些规范有所不同,COM未解决不同语言之间调用,提供了IDispatch接口. 2 ...

  9. SmartRoute之远程接口调用和负载

    基于接口的调用远比基于基础消息交互来得更简单和便于维护,特别在业务展现上,接口作为业务表现更适合其便利性.为了让SmartRoute更适合业务应用集成,在新的一年开始SmartRoute集成了远程接口 ...

随机推荐

  1. AQS源码深入分析之独占模式-ReentrantLock锁特性详解

    本文基于JDK-8u261源码分析 相信大部分人知道AQS是因为ReentrantLock,ReentrantLock的底层是使用AQS来实现的.还有一部分人知道共享锁(Semaphore/Count ...

  2. 循序渐进VUE+Element 前端应用开发(25)--- 各种界面组件的使用(1)

    在我们使用Vue+Element开发前端的时候,往往涉及到很多界面组件的使用,其中很多直接采用Element官方的案例即可,有些则是在这个基础上封装更好利用.更少代码的组件:另外有些则是直接采用第三方 ...

  3. Linux杂谈:进程锁核+实时线程导致的读写锁死循环

    发现问题 公司项目测试的时候,发现运行一段时间后会出现cpu百分之百的情况. 想着可能是哪里出现了死循环,于是打算用gdb跟一下,结果gdb居然无法attach到进程...... 定位问题 查了查去, ...

  4. ETCD核心机制解析

    ETCD整体机制 etcd 是一个分布式的.可靠的 key-value 存储系统,它适用于存储分布式系统中的关键数据. etcd 集群中多个节点之间通过Raft算法完成分布式一致性协同,算法会选举出一 ...

  5. P4683 [IOI2008] Type Printer 打印机

    题意描述 [IOI2008] Type Printer 打印机 几百年前的 IOI 的题目还是很好的呀. 给你一个 诡异的 打印机,它只能用已有的字符来打印,而且必须每一个都用到.(这岂不是活字印刷术 ...

  6. JDK8中的新时间API:Duration Period和ChronoUnit介绍

    目录 简介 Duration Period ChronoUnit 简介 在JDK8中,引入了三个非常有用的时间相关的API:Duration,Period和ChronoUnit. 他们都是用来对时间进 ...

  7. Thinkphp3.2 cms之登陆模块

    <?php /** * Created by dreamcms. * User: Administrator * Date: 2016/9/5 * Time: 17:15 */ namespac ...

  8. 【笔记】nrf52832广播使用--厂商自定义数据应用

    需求: 1)使用蓝牙不停发送ble广播,发送自定义的数据,并每一秒更新自定义数据. 2)设置不同的发射功率.广播间隔.广播名称 1.初始化 使用nordic官方sdk17版本,打开一个ble串口用例. ...

  9. 天啦噜!仅仅5张图,彻底搞懂Python中的深浅拷贝

    Python中的深浅拷贝 在讲深浅拷贝之前,我们先重温一下 is 和==的区别. 在判断对象是否相等比较的时候我们可以用is 和 == is:比较两个对象的引用是否相同,即 它们的id 是否一样 == ...

  10. java小工具,使用Swing展示左树右表结构

    代码直接上: 入口类 import java.io.File; import java.util.ArrayList; import java.util.List; import org.json.J ...