原文地址:http://www.cnblogs.com/hoodlum1980/archive/2009/12/30/1635702.html
COM 是 WIN32 系统中最复杂和晦涩,最重要的技术。
【备注:以下是个人看法】COM 是比传统的 API 提供方式(*.h, *.lib, *.dll) 更”高“层次的服务标准, 从某种意义上说,COM也是一种”API“,但它的使用和实现都要比传统API复杂的多。COM 的宗旨是提供这样的一种标准,使操作系统,独立软件开发商之间以一种标准方式提供交互性服务。COM 相比传统API合作方式相比:两者本质上都提供的二进制代码(编译后产品),前者是基于有组织的,语言中立的 具有面向对象特性的标准。后者是零散的,语言相关,面向过程的方式。前者的显著优点是,它的组件是共享式的安装在系统上的一种服务,组件位置对用户是透明的,因此它仅需要在每个客户端部署一份(甚至部署在一台远程计算机)。而后者位置不透明因此未必能够做到这一点。COM 技术使用推出新接口实现升级,以降低对客户端的影响。而传统 API 提供方式有可能在升级时导致DLL版本混乱。
由于对语言中立性的追求以及COM的标准,COM 的代码相比普通的 C/C++ 可读性更低,但也具有一定规律性。ATL中提供了一些辅助类来降低 COM 代码的使用难度。
这里是两个例子。
【例子1】: 在 Windows Mobile 上获取收件箱中的短信内容。
(1)每个帐户对应的是一个MsgStore。每个MsgStore可包含一组文件夹。
(2)提供EntryID,用 OpenEntry 方法打开MAPI对象。
(3)MAPI容器 使用 GetContentTable 获取容器内内容的描述表。
(4)MAPI Table 使用SetColumns 设定要查询的字段。
(5)使用 MAPI 提供的函数释放相关查询信息分配的内存。
更多细节参考 相关SDK 文档。
下面是代码。当打开收件箱以后,短信是按照时间升序排列的,因此使用 SeekRow 方法把游标游动到最后一行读出既是最近收到的短信。

bool GetLastMessage(TCHAR * buffer, int bufferSize)
{
HRESULT hr = S_OK;
ICEMAPISession * pSession = NULL;
IMsgStore * pMsgStore = NULL;
IMAPIFolder * pFolder = NULL;
IMessage * pMsg = NULL;
// 和Table有关的MAPI对象
IMAPITable * pTable = NULL;
SRowSet * pRows = NULL;
SPropValue * pSProps = NULL;
ULONG ulObjType; // 对象类型
ULONG cCount = 0 ;
int remainLength, subjectLength; // 字符串长度
bool result = false ;
// 设置MsgStore表的要查询字段
SizedSPropTagArray( 2 , Columns1) =
{
2 ,
PR_ENTRYID, // Store的Entry ID
PR_DISPLAY_NAME // Store的Display Name
};
// 设置收件箱内容表要查询的字段
SizedSPropTagArray( 1 , Columns2) =
{
1 ,
PR_ENTRYID // 获取每个消息的EntryID
};
CoInitializeEx(NULL, COINIT_MULTITHREADED);
hr = MAPIInitialize(NULL);
if (FAILED(hr))
{
wcscpy_s(buffer, bufferSize, _T( " failed at: MAPIInitialize " ));
goto TAG_CLEAR;
}
hr = MAPILogonEx( 0 , NULL, NULL, 0 , (LPMAPISESSION * ) & pSession);
if (FAILED(hr))
{
wcscpy_s(buffer, bufferSize, _T( " failed at: MAPILogonEx " ));
goto TAG_CLEAR;
}
hr = pSession -> GetMsgStoresTable( 0 , & pTable);
if (FAILED(hr))
{
wcscpy_s(buffer, bufferSize, _T( " failed at: GetMsgStoresTable " ));
goto TAG_CLEAR;
}
pTable -> SetColumns((LPSPropTagArray) & Columns1, 0 );
while (SUCCEEDED(pTable -> QueryRows( 1 , 0 , & pRows)))
{
if (pRows == NULL || pRows -> cRows != 1 )
break ;
// 开始一条条记录查询,
// pRows->aRow[0].lpProps[0]代表了第一个查询属性,即Entry ID,
// pRows->aRow[0].lpProps[1]则表示Display Name。
if (_tcsicmp(pRows -> aRow[ 0 ].lpProps[ 1 ].Value.lpszW, _T( " SMS " )) == 0 )
{
// 如果是SMS,则获取Store对象
pSession -> OpenEntry(
pRows -> aRow[ 0 ].lpProps[ 0 ].Value.bin.cb, (LPENTRYID)pRows -> aRow[ 0 ].lpProps[ 0 ].Value.bin.lpb,
NULL, MAPI_BEST_ACCESS, & ulObjType,(LPUNKNOWN * ) & pMsgStore
);
break ;
}
FreeProws(pRows);
pRows = NULL;
}
if (pRows != NULL)
{
FreeProws(pRows);
}
if (pTable != NULL)
{
pTable -> Release();
}
if (pMsgStore == NULL)
{
wcscpy_s(buffer, bufferSize, _T( " Cannot Open MsgStore of SMS. " ));
goto TAG_CLEAR;
}
// 查询“收件箱”的ENTRYID
ULONG STags[] = { 1 , PR_CE_IPM_INBOX_ENTRYID };
hr = pMsgStore -> GetProps((SPropTagArray * ) & STags, 0 , & cCount, & pSProps);
if (FAILED(hr))
{
wcscpy_s(buffer, bufferSize, _T( " Cannot Get Inbox's EntryID. " ));
goto TAG_CLEAR;
}
// 打开收件箱文件夹
hr = pMsgStore -> OpenEntry(pSProps[ 0 ].Value.bin.cb, (LPENTRYID)pSProps[ 0 ].Value.bin.lpb,NULL, 0 , & ulObjType, (IUnknown ** ) & pFolder);
MAPIFreeBuffer(pSProps);
if (FAILED(hr))
{
wcscpy_s(buffer, bufferSize, _T( " Cannot Open Folder Of Inbox. " ));
goto TAG_CLEAR;
}
// 获取收件箱的内容表。
pFolder -> GetContentsTable( 0 , & pTable);
// 尝试移动到最后一行(最近收到的短信)
ULONG rowCount;
hr = pTable -> GetRowCount( 0 , & rowCount);
if (FAILED(hr) || rowCount <= 0 )
{
pTable -> Release();
pTable = NULL;
wcscpy_s(buffer, bufferSize, _T( " 收件箱:没有新的消息。 " ));
goto TAG_CLEAR;
}
hr = pTable -> SeekRow(BOOKMARK_BEGINNING, rowCount - 1 , NULL);
hr = pTable -> SetColumns((LPSPropTagArray) & Columns2, 0 );
// 要查询的消息属性:发件人,主题(短信内容)
ULONG pMsgPropTags[] = { 2 , PR_SENDER_EMAIL_ADDRESS, PR_SUBJECT };
while (SUCCEEDED(pTable -> QueryRows( 1 , 0 , & pRows)))
{
// 通过OpenEntry获取IMessage对象
pFolder -> OpenEntry(
pRows -> aRow[ 0 ].lpProps[ 0 ].Value.bin.cb,
(LPENTRYID)pRows -> aRow[ 0 ].lpProps[ 0 ].Value.bin.lpb, // Message's EntryID
NULL,
MAPI_BEST_ACCESS,
& ulObjType,
(LPUNKNOWN * ) & pMsg
);
// 这里拿到每个IMessage对象就代表了Folder里面的每一条Message,可以通过对它的操作来获取想要的信息。
hr = pMsg -> GetProps((SPropTagArray * ) & pMsgPropTags, MAPI_UNICODE, & cCount, & pSProps);
if (SUCCEEDED(hr))
{
if (pSProps[ 0 ].ulPropTag == PR_SENDER_EMAIL_ADDRESS)
{
wcscpy_s(buffer, bufferSize, pSProps[ 0 ].Value.lpszW);
wcscat_s(buffer, bufferSize, _T( " : " ));
}
else
{
buffer[ 0 ] = ' /0 ' ;
}
if (pSProps[ 1 ].ulPropTag == PR_SUBJECT)
{
remainLength = bufferSize - wcslen(buffer) - 1 ;
subjectLength = wcslen(pSProps[ 1 ].Value.lpszW);
if (remainLength >= subjectLength)
wcscat_s(buffer, bufferSize, pSProps[ 1 ].Value.lpszW);
else
{
// 截断字符串,并用三个点表示省略。
wcsncpy_s(buffer + wcslen(buffer), bufferSize, pSProps[ 1 ].Value.lpszW, remainLength - 3 );
wcscat_s(buffer, bufferSize, _T( " ... " ));
}
}
result = true ;
}
else
{
wcscpy_s(buffer, bufferSize, _T( " Cannot Get Message. " ));
}
if (pSProps != NULL)
MAPIFreeBuffer(pSProps);
// 仅仅读取一条Message
break ;
}
if (pRows != NULL)
{
FreeProws(pRows);
pRows = NULL;
}
if (pTable != NULL)
{
pTable -> Release();
pTable = NULL;
}
TAG_CLEAR:
if (pMsg != NULL)
pMsg -> Release();
if (pFolder != NULL)
pFolder -> Release();
if (pMsgStore != NULL)
pMsgStore -> Release();
if (pSession != NULL)
{
pSession -> Logoff( 0 , 0 , 0 );
pSession -> Release();
}
MAPIUninitialize();
CoUninitialize();
return result;
}
本段代码已运用于 LedScreen插件V2.0 中:
【例子2】:在ActiveX控件中调用页面上的 JavaScrpt 方法。
(1)IE调用ActiveX控件的方法,本质上是通过控件的 IDispatch 接口调用的,该技术也就是自动化技术。该技术绑定时间比自定义接口最晚,因此效率不如通过自定义接口调用,但是也最灵活。 当我们在ActiveX 控件中调用页面的脚本时,我们同样通过脚本对象的 IDispatch 接口调用。
(2)假设 ActiveX 控件事先知道页面上的方法的参数列表,(相当于我们已知一个C#委托或者C++函数指针定义),如果不这样假定,则实在意义不大。
(3) 我们在控件接口中添加一个方法,称为InvokeJS。在这个方法里我们试图调用 页面上的一个指定名称的javascript方法。并且我们给这个方法一个字符串参数,内容是”hello world“。代码如下:

STDMETHODIMP CMyControl::InvokeJS(BSTR funName)
{
if (m_document == NULL)
return S_OK;
CComPtr < IHTMLDocument2 > pHtmlDoc;
CComPtr < IDispatch > pScript; // CComPtr:会自动调用Release();
HRESULT hr;
CComBSTR bstrMember;
DISPID dispid; // DISPID即LONG(int32),接口函数的数字序号。
bstrMember.AssignBSTR(funName);
hr = this -> m_document -> QueryInterface(IID_IHTMLDocument2, ( void ** ) & pHtmlDoc);
hr = pHtmlDoc -> get_Script( & pScript);
hr = pScript -> GetIDsOfNames(IID_NULL, & bstrMember, 1 , LOCALE_SYSTEM_DEFAULT, & dispid);
// 调用脚本对象的Invoke方法执行脚本函数:
DISPPARAMS dispparams;
memset( & dispparams, 0 , sizeof dispparams);
dispparams.cArgs = 1 ;
dispparams.rgvarg = new VARIANT[dispparams.cArgs];
for ( int i = 0 ; i < dispparams.cArgs; i ++ )
{
CComBSTR bstr = " hello world " ; // back reading
bstr.CopyTo( & dispparams.rgvarg[i].bstrVal);
dispparams.rgvarg[i].vt = VT_BSTR;
}
dispparams.cNamedArgs = 0 ;
EXCEPINFO excepInfo;
memset( & excepInfo, 0 , sizeof excepInfo);
CComVariant vaResult;
UINT nArgErr = (UINT) - 1 ; // initialize to invalid arg
hr = pScript -> Invoke(dispid, IID_NULL, 0 , DISPATCH_METHOD, & dispparams, & vaResult, & excepInfo, & nArgErr);
return S_OK;
}
浏览器对象实际上我们是在控件的 SetClientSite 方法中获取的, 该方法在创建控件时由容器调用,以给ActiveX控件一个时机去获取一些容器信息。则在MSDN文档中有专门介绍。控件的SetClientSite代码如下:

private :
IWebBrowser2 * m_browser; // 浏览器对象
IDispatch * m_document; // 文档对象
STDMETHODIMP CMyControl::SetClientSite(IOleClientSite * pClientSite)
{
HRESULT hr = S_OK;
IServiceProvider * isp, * isp2 = NULL;
// BSTR url;
if ( ! pClientSite)
{
COMRELEASE(m_browser);
}
else
{
hr = pClientSite -> QueryInterface(IID_IServiceProvider, reinterpret_cast < void **> ( & isp));
if (FAILED(hr))
{
hr = S_OK;
goto cleanup;
}
hr = isp -> QueryService(SID_STopLevelBrowser, IID_IServiceProvider, reinterpret_cast < void **> ( & isp2));
if (FAILED(hr))
{
hr = S_OK;
goto cleanup;
}
hr = isp2 -> QueryService(SID_SWebBrowserApp, IID_IWebBrowser2, reinterpret_cast < void **> ( & m_browser));
if (FAILED(hr))
{
hr = S_OK;
goto cleanup;
}
// 获取文档对象
hr = m_browser -> get_Document( & m_document);
if (FAILED(hr))
{
m_document = NULL;
}
// m_browser->get_LocationURL(&url);
// MessageBox((LPCTSTR)url, _T(""), MB_OK);
cleanup:
// Free resources.
COMRELEASE(isp);
COMRELEASE(isp2);
return hr;
}
return S_OK;
}
(4)假设在页面上具有这样一个JS方法:当我们点击按钮时,页面首先通过IDispatch接口找到控件的InvokeJS方法,然后在控件中又通过IDispatch接口找到脚本的AlertMessage方法,弹出一个MessageBox。

< HEAD >
< TITLE > 对象测试页 </ TITLE >
</ HEAD >
< BODY >
< script type ="text/javascript" >
function AlertMessage(info)
{
alert(info);
}
</ script >
< OBJECT ID ="MyControl" CLASSID ="CLSID:......" ></ OBJECT >
< button onclick ="MyControl.InvokeJS('AlertMessage');" > InvokeJavaScript </ button >
</ BODY >
</ HTML >
【例子3】:通过 ITaskBar 对象隐藏或显示任务栏按钮。

{
HRESULT hr;
ITaskbarList * pTaskbarList;
CoInitialize(NULL);
hr = CoCreateInstance(CLSID_TaskbarList, NULL, CLSCTX_INPROC_SERVER,
IID_ITaskbarList, ( void ** ) & pTaskbarList);
// ============== HrInit();==================
// Initializes the taskbar list object. This method must be called before any other
// ITaskbarList methods can be called.
pTaskbarList -> HrInit();
if (bShow)
{
pTaskbarList -> AddTab(hWnd);
}
else
{
pTaskbarList -> DeleteTab(hWnd);
}
pTaskbarList -> Release();
CoUninitialize();
}
参考内容:
《Mapi实例 》:http://bingqingwu5799.blog.163.com/blog/static/338365512009320111114912/
MSDN & Windows Mobile 6 SDK Documents;
优快云 关于 ActiveX 调用页面脚本的文章。(原连接未记录)