一、创建源对象
1、新建atl项目
2、选择exe,确定
3、在WorkSpace中右键点项目,在弹出菜单中选择NEW ATL OBJECT
4、Category选ojbect,Objects选SimpleObject
5、Names标签中的ShortName中输入Math
6、Attributes标签中选上Support Connection Points
7、在WorkSpace中鼠标右键点击IMath,选择Add Mathod
8、按如下输入
9、打开SourceObj.idl 加入
dispinterface _IMathEvents
{
properties:
methods:
[id(1), helpstring("method Finished")] HRESULT OnSpeak([in] long lResult);
};
10、先build,然后鼠标右键点击CMath,选择Implement Connection Point
再谈出的对话框中,选择_IMathEvents,然后按默认操作。
12、然后
STDMETHODIMP CMath::Speak(long time)
{
// TODO: Add your implementation code here
Fire_OnSpeak(time);
return S_OK;
}
然后再math.h中修改
BEGIN_CONNECTION_POINT_MAP(CMath)
CONNECTION_POINT_ENTRY(DIID__IMathEvents)
END_CONNECTION_POINT_MAP()
然后重新编译
二、创建代理存根DLL
1、在WorkSpace中右键点然后选择Insert Project into Workspace
2、选择SourceObj.mk,
3、选择sourceObj1项目的Project Setting按如下设置
设置好后编译。
4、选择菜单的register control
三、创建接受器
1、右键点击WorkSpace选择add new project
2、接下来按照图1,图2设置生成新的项目,项目名为Sink
3、修改Sink.cpp,代码如下
// Sink.cpp : Implementation of WinMain
// Note: Proxy/Stub Information
// To build a separate proxy/stub DLL,
// run nmake -f Sinkps.mk in the project directory.
#include "stdafx.h"
#include "resource.h"
#include <initguid.h>
#include "Sink.h"
#include "Sink_i.c"
#include "..\SourceObj\SourceObj_i.c"
#include "..\SourceObj\SourceObj.h"
#include <WINDOWS.H>
static _ATL_FUNC_INFO OnSpeakInfo = {CC_STDCALL, VT_EMPTY, 1,{VT_I4}};
static const int DISPID_ONSPEAK = 1;
static const int SPEAK_EVENT_ID = 1;//任意
class CSinkObj : public IDispEventSimpleImpl<SPEAK_EVENT_ID, CSinkObj, &__uuidof(_IMathEvents)>
{
public:
CSinkObj()
{
}
BEGIN_SINK_MAP(CSinkObj)
//Make sure the Event Handlers have __stdcall calling convention
SINK_ENTRY_INFO(SPEAK_EVENT_ID, __uuidof(_IMathEvents), DISPID_ONSPEAK, OnSpeak, &OnSpeakInfo)
END_SINK_MAP()
// Event handler
HRESULT __stdcall OnSpeak(long lResult)
{
Beep(5000,lResult);
return S_OK;
}
};
const DWORD dwTimeOut = 5000; // time for EXE to be idle before shutting down
const DWORD dwPause = 1000; // time to wait for threads to finish up
// Passed to CreateThread to monitor the shutdown event
static DWORD WINAPI MonitorProc(void* pv)
{
CExeModule* p = (CExeModule*)pv;
p->MonitorShutdown();
return 0;
}
LONG CExeModule::Unlock()
{
LONG l = CComModule::Unlock();
if (l == 0)
{
bActivity = true;
SetEvent(hEventShutdown); // tell monitor that we transitioned to zero
}
return l;
}
//Monitors the shutdown event
void CExeModule::MonitorShutdown()
{
while (1)
{
WaitForSingleObject(hEventShutdown, INFINITE);
DWORD dwWait=0;
do
{
bActivity = false;
dwWait = WaitForSingleObject(hEventShutdown, dwTimeOut);
} while (dwWait == WAIT_OBJECT_0);
// timed out
if (!bActivity && m_nLockCnt == 0) // if no activity let´s really bail
{
#if _WIN32_WINNT >= 0x0400 & defined(_ATL_FREE_THREADED)
CoSuspendClassObjects();
if (!bActivity && m_nLockCnt == 0)
#endif
break;
}
}
CloseHandle(hEventShutdown);
PostThreadMessage(dwThreadID, WM_QUIT, 0, 0);
}
bool CExeModule::StartMonitor()
{
hEventShutdown = CreateEvent(NULL, false, false, NULL);
if (hEventShutdown == NULL)
return false;
DWORD dwThreadID;
HANDLE h = CreateThread(NULL, 0, MonitorProc, this, 0, &dwThreadID);
return (h != NULL);
}
CExeModule _Module;
BEGIN_OBJECT_MAP(ObjectMap)
END_OBJECT_MAP()
LPCTSTR FindOneOf(LPCTSTR p1, LPCTSTR p2)
{
while (p1 != NULL && *p1 != NULL)
{
LPCTSTR p = p2;
while (p != NULL && *p != NULL)
{
if (*p1 == *p)
return CharNext(p1);
p = CharNext(p);
}
p1 = CharNext(p1);
}
return NULL;
}
/////////////////////////////////////////////////////////////////////////////
//
extern "C" int WINAPI _tWinMain(HINSTANCE hInstance,
HINSTANCE /*hPrevInstance*/, LPTSTR lpCmdLine, int /*nShowCmd*/)
{
lpCmdLine = GetCommandLine(); //this line necessary for _ATL_MIN_CRT
#if _WIN32_WINNT >= 0x0400 & defined(_ATL_FREE_THREADED)
HRESULT hRes = CoInitializeEx(NULL, COINIT_MULTITHREADED);
#else
HRESULT hRes = CoInitialize(NULL);
#endif
_ASSERTE(SUCCEEDED(hRes));
_Module.Init(ObjectMap, hInstance, &LIBID_SINKLib);
_Module.dwThreadID = GetCurrentThreadId();
TCHAR szTokens[] = _T("-/");
/* int nRet = 0;
BOOL bRun = TRUE;
LPCTSTR lpszToken = FindOneOf(lpCmdLine, szTokens);
while (lpszToken != NULL)
{
if (lstrcmpi(lpszToken, _T("UnregServer"))==0)
{
_Module.UpdateRegistryFromResource(IDR_Sink, FALSE);
nRet = _Module.UnregisterServer(TRUE);
bRun = FALSE;
break;
}
if (lstrcmpi(lpszToken, _T("RegServer"))==0)
{
_Module.UpdateRegistryFromResource(IDR_Sink, TRUE);
nRet = _Module.RegisterServer(TRUE);
bRun = FALSE;
break;
}
lpszToken = FindOneOf(lpszToken, szTokens);
}
if (bRun)
{
_Module.StartMonitor();
#if _WIN32_WINNT >= 0x0400 & defined(_ATL_FREE_THREADED)
hRes = _Module.RegisterClassObjects(CLSCTX_LOCAL_SERVER,
REGCLS_MULTIPLEUSE | REGCLS_SUSPENDED);
_ASSERTE(SUCCEEDED(hRes));
hRes = CoResumeClassObjects();
#else
hRes = _Module.RegisterClassObjects(CLSCTX_LOCAL_SERVER,
REGCLS_MULTIPLEUSE);
#endif
_ASSERTE(SUCCEEDED(hRes));
MSG msg;
while (GetMessage(&msg, 0, 0, 0))
DispatchMessage(&msg);
_Module.RevokeClassObjects();
Sleep(dwPause); //wait for any threads to finish
}*/
IMath *pMath;
HRESULT hr=CoCreateInstance(CLSID_Math,NULL,CLSCTX_LOCAL_SERVER ,__uuidof(IMath),(void**)&pMath);
if(SUCCEEDED(hr))
{
CSinkObj *pSink=NULL;
pSink=new CSinkObj;
pSink->DispEventAdvise(pMath);
pMath->Speak(2000);
if(pSink)
{
if (pSink->m_dwEventCookie != 0xFEFEFEFE)
pSink->DispEventUnadvise(pMath);
delete pSink;
}
}
_Module.Term();
CoUninitialize();
return 0;
}
四、程序流程
Sink调用Source的Speak接口,Speak接口通知Sink,"I am On Speak"
代码到fly.ispace.cn下载