總愛(ài)嘮叨幾句(只關(guān)心技術(shù)的人可以跳過(guò),^_^)
來(lái)北京前,在武漢做開(kāi)發(fā),那里全部用VC,并且采用比較傳統(tǒng)的方法解決問(wèn)題。漸漸的,我感覺(jué)到自己的知識(shí)體系很不完善,做事情總是想采取簡(jiǎn)潔通用完美的方案,然而認(rèn)識(shí)有限總是無(wú)功而返,回到傳統(tǒng)解決途徑上去。那個(gè)地方?jīng)]有人可以教我,可能由于大家專注不同。一個(gè)人苦心鉆研技術(shù),顯得煢煢孑立,很多時(shí)候我都在懷疑為什么別人可以查閱到那么多資料,他們到底有什么秘訣?我看到我的研究生同學(xué),他們都在公司做技術(shù),應(yīng)該很有朝氣活力,敢于研究技術(shù)的。可是我為他們擔(dān)心,他們逐漸淪為代碼機(jī)器,不敢于改革自己現(xiàn)有的編程模式,把思維禁錮在一個(gè)很小的范圍。我的一個(gè)同學(xué)給我發(fā)牢騷,說(shuō)頭又給他一個(gè)MIS項(xiàng)目維護(hù)代碼,抱怨里面代碼如何亂,風(fēng)格如何糟糕,動(dòng)不動(dòng)就死機(jī)。他總問(wèn)我某個(gè)控件實(shí)現(xiàn)特定功能怎么做,我很抱歉,因?yàn)橛械奈乙膊恢馈N蚁肫?年前的自己,熱衷于駕馭控件的技巧,機(jī)器上的各類控件琳瑯滿目,每發(fā)現(xiàn)一個(gè)好控件,如獲珍寶收藏起來(lái),然而大多都沒(méi)有派上用場(chǎng)。學(xué)習(xí)的過(guò)程讓我明白一個(gè)道理:應(yīng)該形成解決問(wèn)題的方案,學(xué)會(huì)解決問(wèn)題才是王道。
大三開(kāi)的COM課程,我剛一接觸,就深深愛(ài)上這個(gè)東西,當(dāng)時(shí)幾個(gè)班就我最用心學(xué),可能大家覺(jué)得很難沒(méi)有認(rèn)真學(xué)。我想說(shuō)的是要感謝COM,它應(yīng)該是C++開(kāi)發(fā)從編碼到設(shè)計(jì)的轉(zhuǎn)折點(diǎn),給我最深的感覺(jué)就是它是一種框架行為。
來(lái)到北京,我的視野開(kāi)闊很多,在一個(gè)技術(shù)遠(yuǎn)遠(yuǎn)超過(guò)自己的人的指點(diǎn)下,我如沐春風(fēng)。我從一個(gè)不怕coding的人成長(zhǎng)成一個(gè)不怕設(shè)計(jì)的人,自然心高氣爽許多。
框架技術(shù)體系一直喜歡講體系,學(xué)習(xí)技術(shù),總是想把握住整個(gè)體系,有利有弊。利就是眼界開(kāi)闊,增加底氣;弊就是戰(zhàn)線太長(zhǎng),自己很疲憊。非常贊成中國(guó)的中庸思想,對(duì)于我來(lái)說(shuō),COM就是中庸,上下開(kāi)壑,開(kāi)辟.NET和OLE學(xué)習(xí)之路。
Automation作為COM之前的一種老技術(shù),翻譯過(guò)來(lái)叫自動(dòng)化。由于一開(kāi)始接觸的就是COM,習(xí)慣COM和MFC分離開(kāi)發(fā),因此一直沒(méi)有很好理解Automation技術(shù),只知道它集成在MFC應(yīng)用中,以類型庫(kù)方式提供服務(wù)。
昨天寫完MFC和VBA集成后,突然感覺(jué)到MFC應(yīng)用和VBE集成之間的牽連,能感覺(jué)到VBE就像一個(gè)提供Automation的Server。于是我花了一些事件研究Automation,不知為何,我總懷戀老技術(shù),喜歡琢磨“過(guò)時(shí)”的東西。
總感嘆Office系列軟件,如此無(wú)縫集成,毫無(wú)疑問(wèn),它的模型是非常清晰的,框架架構(gòu)出來(lái)的應(yīng)該就是模型,這種架構(gòu)技巧需要技術(shù)的積累。我曾經(jīng)下決心學(xué)習(xí)設(shè)計(jì)模式,每每都沉不住氣。當(dāng)我深入學(xué)習(xí)MFC各種應(yīng)用開(kāi)發(fā)模型的時(shí)候,我明白了高人思維為何深邃,源于他們對(duì)MFC的深入探索。MFC在許多人眼里過(guò)時(shí)了,我談過(guò)的很多技術(shù)也有人指出落后了,我喜歡深入研究MFC,因?yàn)樗锩嬗性S多思想值得學(xué)習(xí)。
Automation開(kāi)發(fā)這里我以對(duì)話框應(yīng)用程序討論自動(dòng)化,當(dāng)然象office那樣的大型軟件在app,mainframe,doc層面上都有。
建立一個(gè)基于對(duì)話框的MFC應(yīng)用程序,注意選擇automation支持。簡(jiǎn)單來(lái)說(shuō)實(shí)現(xiàn)了IDispatch接口的類支持自動(dòng)化。
生成一個(gè)對(duì)話框類和一個(gè)對(duì)話框代理類,VBA里面也存在這種思想。一個(gè)C++類對(duì)象是死的,接口是活得,所以要是的應(yīng)用靈活,就必須為對(duì)象配備一個(gè)組件對(duì)象,外界可以通過(guò)組件接口操縱C++對(duì)象。
Dlg類中有成員
CAutomationServerDlgAutoProxy*?m_pAutoProxy;代理類中有成員
CAutomationServerDlg*?m_pDialog;這個(gè)就稱為“對(duì)偶”設(shè)計(jì)模式吧!
這個(gè)程序可以作為獨(dú)立程序啟動(dòng),也可以作為自動(dòng)化服務(wù)啟動(dòng)。
為自動(dòng)化接口添加屬性,為自動(dòng)化控制(客戶程序)提供服務(wù):

自動(dòng)化屬性實(shí)現(xiàn):
BOOL?CAutomationServerDlgAutoProxy::GetVisible()?


{
????//?TODO:?Add?your?property?handler?here
????
????return?(m_pDialog!=NULL?&&?(m_pDialog->GetStyle()&WS_VISIBLE)!=0);
}

void?CAutomationServerDlgAutoProxy::SetVisible(BOOL?bNewValue)?


{
????//?TODO:?Add?your?property?handler?here
????if(bNewValue?=?GetVisible())

????
{
????????return;
????}
????//?visiable
????if(bNewValue)

????
{
????????//?embed?start
????????if(m_pDialog?==?NULL)

????????
{
????????????if(AfxGetThread()->m_pMainWnd?==?NULL)

????????????
{
????????????????m_pDialog?=?new?CAutomationServerDlg();
????????????????AfxGetThread()->m_pMainWnd?=?m_pDialog;
????????????}
????????????//?show?it
????????????m_pDialog->ShowWindow(SW_SHOWNORMAL);
????????}
????}
????else

????
{
????????m_pDialog->ShowWindow(SW_HIDE);
????}
}
BSTR?CAutomationServerDlgAutoProxy::GetWindowText()?


{
????CString?strResult;
????//?TODO:?Add?your?property?handler?here
????m_pDialog->GetWindowText(strResult);
????return?strResult.AllocSysString();
}

void?CAutomationServerDlgAutoProxy::SetWindowText(LPCTSTR?lpszNewValue)?


{
????//?TODO:?Add?your?property?handler?here
????m_pDialog->SetWindowText(lpszNewValue);
} 自動(dòng)化注冊(cè)機(jī)制
在自動(dòng)化代理類里面有這樣一個(gè)宏聲明:
?
IMPLEMENT_OLECREATE2(CAutomationServer2DlgAutoProxy,?"AutomationServer2.Application",?0xaa98c00e,?0xd46f,?0x4e9e,?0xaf,?0x97,?0xc8,?0x24,?0xa0,?0x29,?0xe4,?0xa4)

看看
IMPLEMENT_OLECREATE2的定義:
#ifndef?IMPLEMENT_OLECREATE2
#define?IMPLEMENT_OLECREATE2(class_name,?external_name,?l,?w1,?w2,?b1,?b2,?b3,?b4,?b5,?b6,?b7,?b8)?\
????AFX_DATADEF?COleObjectFactory?class_name::factory(class_name::guid,?\
????????RUNTIME_CLASS(class_name),?FALSE,?_T(external_name));?\
????const?AFX_DATADEF?GUID?class_name::guid?=?\

????????
{?l,?w1,?w2,?
{?b1,?b2,?b3,?b4,?b5,?b6,?b7,?b8?}?};
#endif?//?IMPLEMENT_OLECREATE2

具體注冊(cè)代碼是:
COleObjectFactory::UpdateRegistryAll(); 但是這個(gè)函數(shù)似乎與上面的宏定義沒(méi)有關(guān)系,為了找到關(guān)系我進(jìn)入U(xiǎn)pdateRegistryAll函數(shù)內(nèi)部,發(fā)現(xiàn):
for?(COleObjectFactory*?pFactory?=?pModuleState->m_factoryList;
????????pFactory?!=?NULL;?pFactory?=?pFactory->m_pNextFactory)

????
{
????????if?(!pFactory->UpdateRegistry(bRegister))

????????
{
????????????AfxUnlockGlobals(CRIT_OBJECTFACTORYLIST);
????????????return?FALSE;
????????}
????} 原來(lái)模塊狀態(tài)記錄了所有類廠列表,為了找到類廠列表加入類廠位置,我搜索了mfc源碼,發(fā)現(xiàn)OLEFACT.cpp里面有這樣的代碼:
COleObjectFactory::COleObjectFactory(REFCLSID?clsid,
????CRuntimeClass*?pRuntimeClass,?BOOL?bMultiInstance,?LPCTSTR?lpszProgID)


{
????ASSERT(pRuntimeClass?==?NULL?||
????????pRuntimeClass->IsDerivedFrom(RUNTIME_CLASS(CCmdTarget)));
????ASSERT(AfxIsValidAddress(&clsid,?sizeof(CLSID),?FALSE));
????ASSERT(lpszProgID?==?NULL?||?AfxIsValidString(lpszProgID));

????//?initialize?to?unregistered?state
????m_dwRegister?=?0;???//?not?registered?yet
????m_bRegistered?=?FALSE;
????m_clsid?=?clsid;
????m_pRuntimeClass?=?pRuntimeClass;
????m_bMultiInstance?=?bMultiInstance;
????m_lpszProgID?=?lpszProgID;
????m_bOAT?=?(BYTE)?OAT_UNKNOWN;

????//?licensing?information
????m_bLicenseChecked?=?FALSE;
????m_bLicenseValid?=?FALSE;

????//?add?this?factory?to?the?list?of?factories
????m_pNextFactory?=?NULL;
????AFX_MODULE_STATE*?pModuleState?=?_AFX_CMDTARGET_GETSTATE();
????AfxLockGlobals(CRIT_OBJECTFACTORYLIST);
????pModuleState->m_factoryList.AddHead(this);
????AfxUnlockGlobals(CRIT_OBJECTFACTORYLIST);

????ASSERT_VALID(this);
} 原來(lái)類廠構(gòu)造函數(shù)里面把自己加入到模塊狀態(tài)的列表里面。
現(xiàn)在總結(jié)一下:IMPLEMENT_OLECREATE2宏定義了一個(gè)類廠實(shí)例,在構(gòu)造的時(shí)候加入模塊狀態(tài)的類廠列表,然后在模塊初始化的時(shí)候,調(diào)用類廠的UpdateRegistry函數(shù)更新注冊(cè)表,實(shí)現(xiàn)注冊(cè)。
Automation調(diào)用下面生成一個(gè)客戶程序控制AutomationServer,基于MFC對(duì)話框應(yīng)用程序。
添加自動(dòng)化程序產(chǎn)生的類型庫(kù):添加類->從類型庫(kù)導(dǎo)入剛生成的tlb文件,向?qū)?huì)生成調(diào)用類。
創(chuàng)建自動(dòng)化組件,顯示自動(dòng)化服務(wù)程序:
if(m_ias.CreateDispatch("AutomationServer.Application"))


{
????m_ias.SetVisible(TRUE);
}通過(guò)按鈕事件設(shè)置服務(wù)器窗口標(biāo)題,演示調(diào)用服務(wù):
void?CAutomationControlDlg::OnButtonSettext()?


{
????//?TODO:?Add?your?control?notification?handler?code?here
????UpdateData();
????if(m_ias.m_lpDispatch)

????
{
????????m_ias.SetWindowText(m_strText);
????}
}

void?CAutomationControlDlg::OnButtonGettext()?


{
????//?TODO:?Add?your?control?notification?handler?code?here
????if(m_ias.m_lpDispatch)

????
{
????????m_strText?=?m_ias.GetWindowText();
????????UpdateData(FALSE);
????}
} 下面是截圖控制啟動(dòng)激活自動(dòng)化服務(wù)程序:

調(diào)用自動(dòng)化服務(wù)功能
代碼下載
posted on 2006-08-31 20:33
萬(wàn)連文 閱讀(4376)
評(píng)論(7) 編輯 收藏 引用 所屬分類:
MFC