COM入门(2)

1. 实现COM组件的自注册和反注册
实现COM组件的自注册和反注册,本质上就是写注册表与删注册表。需要在DLL中引出两个函数:DllRegisterServer和DllUnregisterServer,让这两个函数实现注册表操作。
LPCTSTR RegTable[][ 3 ] =
{
    {L
" CLSID\\{586CDC7B-09F1-4f44-A110-F0E604AED81E} " , 0 , L " BeginningCOM " },
    {L
" CLSID\\{586CDC7B-09F1-4f44-A110-F0E604AED81E}\\InprocServer32 " , 0 , (LPCTSTR) - 1 },
    {L
" CLSID\\{586CDC7B-09F1-4f44-A110-F0E604AED81E}\\InprocServer32 " , L " ThreadingModel " , L " Both " }
};

STDAPI DllUnregisterServer()
{
    HRESULT hr
= S_OK;

    
int regCount = sizeof (RegTable) / sizeof ( * RegTable);

    
for ( int i = regCount - 1 ; i >= 0 ; i -- )
    {
        LSTATUS error
= RegDeleteKey(HKEY_CLASSES_ROOT, RegTable[i][ 0 ]);

        
if (error != ERROR_SUCCESS)
        {
            hr
= S_FALSE;
        }
    }

    
return hr;
}

STDAPI DllRegisterServer(
void )
{
    HRESULT hr
= S_OK;
    TCHAR szFileName[MAX_PATH];

    ZeroMemory(
& szFileName, MAX_PATH * sizeof (TCHAR));

    GetModuleFileName(g_hModule, szFileName, MAX_PATH);

    
int regCount = sizeof (RegTable) / sizeof ( * RegTable);

    
for ( int i = 0 ; i < regCount; i ++ )
    {
        
if (RegTable[i][ 2 ] == (LPCTSTR) - 1 )
        {
            RegTable[i][
2 ] = szFileName;
        }

        HKEY hKey;
        LSTATUS error
= ::RegCreateKey(HKEY_CLASSES_ROOT, RegTable[i][ 0 ], & hKey);

        
if (error == ERROR_SUCCESS)
        {
            error
= RegSetValueEx(hKey, RegTable[i][ 1 ], 0 , REG_SZ, ( const BYTE * )RegTable[i][ 2 ], (lstrlen(RegTable[i][ 2 ]) + 1 ) * sizeof (TCHAR));
            RegCloseKey(hKey);
        }

        
if (error != ERROR_SUCCESS)
        {
            DllUnregisterServer();
        }
    }

    
return hr;
}

现将需要修改的注册表内容放到一个二维数组中,第一维内容分别是注册表键,项,值。-1为占位符,标识该位置放置COM组件的路径,在写注册表之前将其替换。
删注册表时,应该先删子键,否则删除会失败。
同时需要在BeginningCOM.def中导出这两个函数:
LIBRARY     " BeginningCOM "
EXPORTS
    DllGetClassObject
private
    DllRegisterServer
private
    DllUnregisterServer
private

这样就可以用regsvr32.exe注册和反注册COM组件。

2. 自定义接口
可以用IDL文件定义,也可以在代码中直接继承IUnknown接口实现。用IDL定义接口的好处是,MIDL工具能自动生成GUID描述,接口存根/代理等。IDL全称为:Interface Definition Language,其语法类似于C语言,去掉了存在二义性的内容,并进行了扩展。作为示例,我们定义一个IBeginningCOM的接口,里面添加一个Sum函数和一个Num属性。
import " oaidl.idl " ;
import
" ocidl.idl " ;

[
object , uuid(93C3840F - AD5A - 4020 - AAAB - 313C4B61B184)]
interface IBeginningCOM : IUnknown
{
    HRESULT Sum([
in ] int a, [ in ] int b, [ out , retval] int * sum);
    [propget] HRESULT Num([
out , retval] int * pVal);
    [propput] HRESULT Num([
in ] int val);
}

[
    uuid(D9161D4D
- 66C0 - 4ae6 - 9264 - C322BDE034C7),
    version(
1.0 ),
    helpstring(
" BeginningCOMLib " )
]
library BEGINNINGCOMLib
{
    importlib(
" stdole32.tlb " );
    importlib(
" stdole2.tlb " );

    [
        uuid(586CDC7B
- 09F1 - 4f44 - A110 - F0E604AED81E),
        helpstring(
" BeginningCOM Lib " )
    ]
    coclass BeginningCOM
    {
        [
default ] interface IBeginningCOM;
    };
};

引入的oaidl.idl等包含了系统标准接口声明等,直接引入即可。接口的object和接口的名字这两个特性是必须的,为防止接口重名,用GUID来表示接口的名字。propget和proppub表示方法映射到属性,如VB中,在不支持属性的语言,如C++,映射为get_XXX,put_XXX。每个IDL文件只能有一个library标识,内部可以有多个coclass。
IDL文件经MIDL工具编译后,生成四个文件:
*_h.h        接口说明文件
*_i.c        GUID描述文件
*_p.c        接口存根/代理实现文件
dlldata.c    包含存根/代理相关内容

// BeginningCOM继承自IBeginningCOM接口
// ...
class BeginningCOM : public IBeginningCOM
// ...

// 实现IBeginningCOM接口成员,属性需要分别实现set/put。
// ...
STDMETHODIMP BeginningCOM::Sum( int a, int b, int * sum)
{
    
* sum = a + b;
    
return S_OK;
}

STDMETHODIMP BeginningCOM::get_Num(
int * pVal)
{
    
* pVal = m_Num;
    
return S_OK;
}

STDMETHODIMP BeginningCOM::put_Num(
int val)
{
    m_Num
= val;
    
return S_OK;
}

3. 实现IClassFactory接口
IClassFactory是系统定义的对象创建的标准接口。IClassFactory有两个方法:LockServer和CreateInstance。LockServer用于进程外激活时,COM内部对其进行调用,目前我们都是进程内激活,可先不实现;CreateInstance用于创建请求的类对象,该方法的第一个参数在聚合时使用。
实现IClassFactory:
#pragma once
#include
" unknwn.h "

class BeginningCOMClassFactory :
    
public IClassFactory
{
public :
    BeginningCOMClassFactory(VOID);    
// IUknown
    STDMETHODIMP QueryInterface(REFIID riid, VOID ** ppv);
    STDMETHODIMP_(ULONG) AddRef(VOID);
    STDMETHODIMP_(ULONG) Release(VOID);

    
// IClassFactory
    STDMETHODIMP CreateInstance(LPUNKNOWN punkOuter, REFIID riid, VOID ** ppv);
    STDMETHODIMP LockServer(BOOL fLock);

protected :
    ULONG m_ulRefCount;
};

// ...
STDMETHODIMP BeginningCOMClassFactory::CreateInstance(LPUNKNOWN punkOuter, REFIID riid, VOID ** ppv)
{
    BeginningCOM
* pbc;
    HRESULT hr;

    
// 暂不支持聚合
     if (punkOuter != NULL)
    {
        
return CLASS_E_NOAGGREGATION;
    }

    pbc
= new BeginningCOM;

    
if (pbc == NULL)
    {
        
return E_OUTOFMEMORY;
    }

    pbc
-> AddRef();
    hr
= pbc -> QueryInterface(riid, ppv);
    pbc
-> Release();

    
return hr;
}

STDMETHODIMP BeginningCOMClassFactory::LockServer(BOOL fLock)
{
    
return E_FAIL;
}

DllGetClassObject返回是否实现IClassFactory即可。
STDAPI DllGetClassObject(REFCLSID rclsid, REFIID riid,
void ** ppv)
{
    
if (rclsid == CLSID_BeginningCOM)
    {
        BeginningCOMClassFactory
* pbc = new BeginningCOMClassFactory;

        
if (pbc == NULL)
        {
            
return E_OUTOFMEMORY;
        }

        
return pbc -> QueryInterface(riid, ppv);
    }

    
* ppv = 0 ;

    
return CLASS_E_CLASSNOTAVAILABLE;
}

实现了IClassFactory接口,创建COM对象的代码可以通过查询IClassFactory接口,然后调用该接口的CreateInstance方法:
HRESULT hr = NULL;
IClassFactory
* pcf;
IBeginningCOM
* pbc;

hr
= CoGetClassObject(CLSID_BeginningCOM, CLSCTX_INPROC_SERVER, NULL, IID_IClassFactory, ( void ** ) & pcf);

if (SUCCEEDED(hr))
{
    hr
= pcf -> CreateInstance(NULL, IID_IBeginningCOM, ( void ** ) & pbc);
// ...

也可以用CoCreateInstance创建,该函数包装了上面的两布,在分布式环境下能减少一次客户端与服务器之间的通信。
// ...
hr = CoCreateInstance(CLSID_BeginningCOM, NULL, CLSCTX_INPROC_SERVER, IID_IBeginningCOM, ( void ** ) & pbc);
// ...
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
第一部分 了解COM 第1章 COM概述 何谓CoM COM术语 COM利与弊 COM的好处 COM的局限性 COM组件与接口 何谓接口 接口特征 接口类型 接口规则 接口设计 COM组件的实现规则 实现IUnknown规则 内存管理规则 引用计数规则 COM激活 COM类型 COM客户机 COM服务器 ActiveX控件 COM与面向对象技术 包装 抽象 多态 继承 COMTrader应用程序 小结 第2章 由VC++建立并使用COM服务器 IDL文件 建立第一个COM服务器 定义自定义接口 实现IUnknown和自定义接口 完成COM服务器 生成测试客户机 用ATL建立COM服务器 关于ATL 用ATL建立进程内COM服务器 用ATL建立进程外COM服务器 线程与COM服务器 Win32多线程应用 线程COM组件 自动化与IDispatch 用VC++实现IDispatch ATL与自动化 Automation数据类型 再谈类型库 C++自动化客户机 VB自动化客户机 小结 第3章 用VB建立并使用COM服务器 选择COM项目 设计接口 描述接口 浏览接口 生成对象 使用ClassBuilder 增加属性 增加方法 增加事件与枚举 使用ActiveXDataObject(ADO) 在服务器组件中使用Recordset对象 在客户机组件中使用ADOR 生成断开的Recodset 生成自己的RecodsctS 使用用户定义类型 错误处理 服务器客户机错误处理 使用VBErr.Raise机制 在VB中使用线程模型 设置线程模型 了解再入性与公寓 小结 第二部分 COM与Internet 第4章 在VC++中建立并使用ActiveX控件 ACtiveX控件概还 属性与方法 控件与容器通信 事件与连接点 建立第一个控件 生成控件 测试控件 增加方法 增加属性 增加事件 增加属性页 允许属性保持 使用控件 建立复合控件 增加复合控件 增加功能 增加事件 处理复合控件事件 处理错误 使用控件 小结 第5章 在VB中建立并使用ActiveX控件 VB控件简介 约束与无约束控件 控件生成技术 属性类型 方法 属性配置 过程属性 环境属性配置 运行时只读属性 只在运行时有效的属性 扩展属性 容器属性 合成控件属性 可关联属性 持续与属性包 属性包 使用ActiveX控件界面向导 了解控件寿命 生成ActiveX控件 生成无约束控件 生成设计时数据约束控件 生成运行数据约束控件 小结 第6章 用VC++建立InternetCOM组件 IEActiveX控件 轻量级控件 安全控件 持续属性 文档对象模型编程 活动服务器组件 活动服务器页面 ASP页面的COM组件 小结 第7章 用VB建立InternetCOM组件 无窗口控件 ActiveX控件容器的线程模型 ActiveX控件的安全性 Web页面访问 VBDHTML项目 DHTML项目基础 DHTML应用程序样本 VBIIS应用程序 WebClass 一个IIS应用程序样本 设计控件 设计控件与HTML文件 样本设计控件 小结 第三部分 了解DCOM 第8章 DCOM概述 何谓DCOM 为什么使用DCOM DCOM操作 DCOM组件位置 进程内或进程外组件 代理 RPC(RemoteProcedureCall,远程过程调用) 调动 数据传递 DCOM配置实用程序 DCOM应用程序的安全机制 验证 授权 加密 整性检查 小结 第9章 用VC++建立DCOM服务器 标准与自定义调动 标准调动 自定又调动 网络通伯 远程激活 AppID注册表项 可配置AppID注册表项参数 IUknown优化 DCOM与NT服务 NT服务解剖 基于NT服务的COM服务器 小结 第10章 用VB建立DCOM服务器 应用程序对象模型 何谓对象模型 如何生成对象模型 DCOM设计准则与技术 再论调动 按数值与按引用 DCOM进程外服务器 建立DCOM组件 增加测试客户机 IIS应用程序 增加WebClasses 使用模板 增加自定义Webltems 远程错误处理 小结 第四部分 了解COM++ 第11章 COM++概述 COM与WindowsDNA 用户界面层技术 中间层技术 数据库层技术 组件服务配置 事务处理 排队组件(QC) 实时结构的限制 事务性消息排队 排队组件结构 排队组件故障恢复 QC安全性 动态负荷平衡 对象地 小结 第12章 用VC++建立COM++组件 ADO编程 ADO与OLEDB VC++中的ADO VC++的ADO扩展 建立COM++应用程序 温习IObjectContext接口 用ATL建立COM++组件 编制基于角色的安全性 处理COM+事务 控制事务结果 指定事务属性 确定事务情境 传递接口指针 共享状态 建立事务性COM+组件 小结 第13章 用VB建立COM+组件 了解事务 事务与多层应用程序 COM+与事务 事务属性:ACID COM+系统简介 COM+运行环境 COMComponentServices COM+接口 资源分配器 应用程序组件 探索COM+编程模型 COM+组件作为COMDLL 基本COM+编程规则 COM+API 用VB编程COM+ 对象描述表 COM+组件的生命周期 ObjectControl接口 MTS活动 COM+中生成对象 安全引用 组件之间的参数传递 数据类型 使用分布式事务 分布式事务协调器(MSDTC) COM+事务的工作 事务与有状态对象 使用共享属性管理器(SPMSharedProperyManager) 小结 第14章 了解MSMQ 何谓MSMQ MSMQ的好处 MSMQ组件 队列 消息 MSMQ对象模型 MSMQ设置 MSMQ基础 消息发送 消息接收 MSMQ事件 MSMQ事务 小结 第五部分 高级COMCOM+ 第15章 VC++与VB中的COM+服务 了解COM+激活 描述表包装器 激活顺序 使用即时(JIT)激活 使用对象构造 中性公寓简介 了解同步域 表示事务状态 取得对象信息 使用对象地 对象池的好处 对象地要求 对象地配置 使用排队组件 QC限制 QC配置 QC调用 QC播放控件 使用负荷平衡 负荷平衡要求 负荷平衡配置 小结 第16章 COMCOM+安全性 何谓安全性 WindowsNT安全简介 NT验证 NT扮演 NT访问控制 COM安全结构 验证 访问控制 启动权限 标_ 扮演与掩盖 安全总括 COM+安全 COM+说明性安全 COM+角色 编程COMCOM+安全 整个进程安全 接口级安全 激活安全 服务器方安全 调用描述表安全信息 SecuntyProperty信息 安全性与数据库访问 小结 第17章 Windows2000中的新COM特性 同步机制 COM同步API COM同步接口 异步COM 异步接口构造 异步接口调用 关于异步服务器与客户机 让服务器进行异步处理 调用序列化与自动完成 COM管道 COM管道接口 异步管道与提前读取 调用对象与调用取消 调用取消请求 调用取消处理 轻量级处理器 标准LWH 自定义LWH 小结 第六部分 调试与部署COMCOM+应用程序 第18章 调试与剖析COMCOM+应用程序 调试VB组件 调试MTS组件 调试COM+组件 使用条件编译 调试VC++组件 用VisualStUdioAnalyzer剖析 小结 第19章 部署COMCOM+应用程序 DCOM应用程序部署 配置DCOM服务器 配置DCOM客户机 在Internet上部署 Internet上部署与包装 签名CAB文件 许可ActiveX控件 自动化COM+配置 使用COMAdmin接口与集合 配置COM+应用程序 配置组件 配置角色 部署COM+应用程序 小结
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值