探秘Visual Studio 2010中的災(zāi)難恢復(fù)功能
為了提高用戶體驗(yàn),Microsoft在Windows Vista系統(tǒng)中首先引入了重啟管理器(Restart Manager)。它可以幫助應(yīng)用程序維護(hù)其當(dāng)前運(yùn)行狀態(tài),當(dāng)軟件更新后需要重新啟動(dòng),或者是遇到非常嚴(yán)重的問(wèn)題崩潰后,可以重新啟動(dòng)軟件并且恢復(fù)到軟件的當(dāng)前工作狀態(tài)。更重要的是,它還可以恢復(fù)自動(dòng)保存的軟件數(shù)據(jù)狀態(tài),盡量保證用戶數(shù)據(jù)的安全。有了重啟管理器,軟件就可以很快地從災(zāi)難中恢復(fù)過(guò)來(lái),實(shí)現(xiàn)快速“災(zāi)后重建”。
圖1 Visual Studio 2010的重啟管理
重啟管理器主要應(yīng)用在下面兩個(gè)方面:
◆軟件更新
很多時(shí)候,軟件或者操作系統(tǒng)升級(jí)后,需要重新啟動(dòng)才可以生效。在這種情況下,我們就可以使用重啟管理器自動(dòng)關(guān)閉真正運(yùn)行的軟件,然后進(jìn)行更新,更新完成后自動(dòng)重新啟動(dòng)軟件,并且恢復(fù)到軟件當(dāng)前的工作狀態(tài)。這將使得軟件的更新更加流暢和智能。
◆軟件災(zāi)難恢復(fù)
當(dāng)軟件遇到嚴(yán)重錯(cuò)誤,進(jìn)程崩潰的時(shí)候,可以使用重啟管理器重新啟動(dòng)軟件,恢復(fù)軟件自動(dòng)保持的數(shù)據(jù),讓軟件可以快速地從災(zāi)難中恢復(fù)過(guò)來(lái)。
為了支持重啟管理器,微軟提供了一套R(shí)estart Manassas API函數(shù)來(lái)完成這些工作。這些函數(shù)定義在
◆RMStartSession
創(chuàng)建一個(gè)新的重啟任務(wù)。
◆RMGetList
這個(gè)函數(shù)可供安裝程序使用,它可以得到所有被影響的應(yīng)用程序及其當(dāng)前狀態(tài)。
◆RMRegisterResources
注冊(cè)重啟任務(wù)的資源,例如文件名,服務(wù)或者是RM_UNIQUE_PROCESS結(jié)構(gòu)體。
◆RMRestart
重新啟動(dòng)被RmShutdown關(guān)閉的應(yīng)用程序或者服務(wù),當(dāng)然,這些應(yīng)用程序或者服務(wù)都需要通過(guò)RegisterApplicationRestart事先進(jìn)行注冊(cè)。
◆RMShutDown
關(guān)閉應(yīng)用程序或者服務(wù)。
◆RMEndSession
結(jié)束重啟任務(wù)。
添加對(duì)重啟管理器的支持
雖然Windows Vista系統(tǒng)本身提供了對(duì)重啟管理器的支持,但是對(duì)于應(yīng)用程序本身,也同樣需要一些額外的工作,以完成對(duì)重啟管理器的支持。
對(duì)于新創(chuàng)建的MFC應(yīng)用程序,我們可以簡(jiǎn)單地在“MFC應(yīng)用程序向?qū)А敝性O(shè)置是否需要支持重啟管理器。
圖2 MFC應(yīng)用程序向?qū)?/P>
在“MFC應(yīng)用程序向?qū)А钡母呒?jí)特性選項(xiàng)卡中有關(guān)于重啟管理器的選項(xiàng)。其中,如果僅僅選中“Support Restart Manager”選項(xiàng),表示你的應(yīng)用程序?qū)H僅支持重新啟動(dòng)。換句話說(shuō),你的應(yīng)用程序可以在升級(jí)或者崩潰之后重新啟動(dòng),但是無(wú)法自動(dòng)打開(kāi)未關(guān)閉的文檔,無(wú)法對(duì)數(shù)據(jù)進(jìn)行恢復(fù)。
如果同時(shí)選中“Reopen previously open documents”選項(xiàng),它表示你的應(yīng)用程序可以在重啟之后重新打開(kāi)之前打開(kāi)的文檔,也就是自動(dòng)恢復(fù)到當(dāng)前的工作狀態(tài)。
如果選中了“Support application recover”選項(xiàng),它表示你的應(yīng)用程序在重新啟動(dòng)后,不僅可以重新打開(kāi)之前打開(kāi)的文檔,還會(huì)嘗試恢復(fù)自動(dòng)保存的文檔。它將彈出一個(gè)任務(wù)對(duì)話框(Unicode版本)或者消息框(非Unicode版本),詢問(wèn)用戶是否需要恢復(fù)自動(dòng)保持的文檔。如果用戶選擇“Yes”,那么自動(dòng)保持的文檔將被打開(kāi)作為當(dāng)前文檔。如果用戶選擇“No”,那么用戶***保存的文檔將被打開(kāi)作為當(dāng)前文檔,同時(shí)自動(dòng)保存的文檔將被刪除。
這里需要注意的是,只有文檔視圖類型的應(yīng)用程序才支持“Reopen previously open documents”選項(xiàng)和“Support application recover”選項(xiàng),對(duì)話框類型的應(yīng)用程序只支持“Support Restart Manager”選項(xiàng)。
對(duì)于Visual Studio 2010中新創(chuàng)建的MFC應(yīng)用程序,可以在應(yīng)用程序向?qū)е羞M(jìn)行設(shè)置,添加對(duì)重啟管理器的支持。那么對(duì)于很多已有的MFC應(yīng)用程序,如果同樣想獲得重啟管理器的支持,應(yīng)該怎么辦呢?實(shí)際上,對(duì)于已有的MFC應(yīng)用程序,要想獲得重啟管理器的支持很簡(jiǎn)單。在新版本的MFC中,CWinApp類增加了一個(gè)新的成員變量,用于控制應(yīng)用程序?qū)χ貑⒐芾砥鞯闹С?,我們只需要在?yīng)用程序的構(gòu)造函數(shù)中,添加幾行代碼,按照我們的需求對(duì)其合理的初始化就可以了。
#p#
跟我們?cè)谏厦嫠榻B的“MFC應(yīng)用程序向?qū)А敝械倪x項(xiàng)相一致,如果你選擇的是“Support Restart Manager”,你可以在初始化函數(shù)中添加如下的代碼:
m_dwRestartManagerSupportFlags = |
如果你想選擇“Reopen previously open documents”,可以添加如下的代碼:
m_dwRestartManagerSupportFlags = |
m_dwRestartManagerSupportFlags = |
CRestartManagerDemoApp::CRestartManagerDemoApp() |
實(shí)例:創(chuàng)建支持重啟管理器的MFC應(yīng)用程序
下面我們以一個(gè)實(shí)際的例子,來(lái)看看如何在我們的MFC應(yīng)用程序中添加對(duì)重啟管理器的支持。
首先,啟動(dòng)Visual Studio 2010 CTP,創(chuàng)建一個(gè)單文檔的應(yīng)用程序RestartManagerDemo。按照我們前面的介紹,在“MFC應(yīng)用程序向?qū)А敝羞x擇“Support Restart Manager”和“Reopen previously open documents”選項(xiàng),以支持應(yīng)用程序的重新啟動(dòng)和文檔的重新打開(kāi)。
為了驗(yàn)證重啟管理器重新打開(kāi)文檔的功能,我們?cè)谖臋n中添加一些數(shù)據(jù),這些數(shù)據(jù)將在程序重新啟動(dòng)后自動(dòng)被加載進(jìn)來(lái)。
// 泡泡類,用于在視圖中顯示圓圈泡泡
class CBubble
{
public:
CBubble(CPoint cp, double fR)
{
m_nCenterPoint = cp;
m_fR = fR;
};
CBubble()
{};
// 圓心
CPoint m_nCenterPoint;
// 半徑
double m_fR;
};class CRestartManagerDemoDoc : public CDocument
{
protected: // create from serialization only
CRestartManagerDemoDoc();
DECLARE_DYNCREATE(CRestartManagerDemoDoc)// Attributes
public:
// 保存數(shù)據(jù)的數(shù)組
CArraym_Array;
// Operations
public:
CArray& GetBubbleArray()
{
return m_Array;
};
//…
};
然后,我們需要實(shí)現(xiàn)文檔的序列化函數(shù),使得我們的文檔數(shù)據(jù)能夠保存和重新加載:
// CRestartManagerDemoDoc serializationvoid CRestartManagerDemoDoc::Serialize(CArchive& ar)
{
// 保存數(shù)據(jù)
if (ar.IsStoring())
{
// TODO: add storing code here
int nSize = m_Array.GetSize();
ar<for(int nIndex = 0; nIndex < nSize; ++nIndex )
{
CBubble tempBubble = m_Array.GetAt( nIndex );
ar<ar< }
}
else // 加載數(shù)據(jù)
{
// TODO: add loading code here
int nSize = 0;
ar>>nSize;
for(int nIndex = 0; nIndex < nSize; ++nIndex )
{
//CBubble tempBubble = m_Array.GetAt( nIndex );
CPoint tempPoint;
double tempR;
ar>>tempPoint;
ar>>tempR;
m_Array.Add( CBubble( tempPoint, tempR) );
}
}
}
完成文檔類的工作后,我們就有了保存數(shù)據(jù)的容器,現(xiàn)在我們需要對(duì)數(shù)據(jù)進(jìn)行修改和顯示。在視圖類中,我們通過(guò)鼠標(biāo)點(diǎn)擊,修改文檔中的數(shù)據(jù),向其中添加CBubble對(duì)象。
void CRestartManagerDemoView::OnLButtonDown(UINT nFlags, CPoint point)
{
// TODO: 在此添加消息處理程序代碼和/或調(diào)用默認(rèn)值
CRestartManagerDemoDoc* pDoc = GetDocument();
ASSERT_VALID(pDoc);
if (!pDoc)
return;
CArray& m_Array = pDoc->GetBubbleArray();
// 以當(dāng)前鼠標(biāo)點(diǎn)擊點(diǎn)為圓心,隨機(jī)半徑構(gòu)造一個(gè)CBubble對(duì)象,并添加到文檔中
m_Array.Add( CBubble( point, rand()%30 ));// 更新視圖顯示
Invalidate();CView::OnLButtonDown(nFlags, point);
}然后,我們將這些數(shù)據(jù)在視圖中顯示出來(lái):
void CRestartManagerDemoView::OnDraw(CDC* pDC)
{
CRestartManagerDemoDoc* pDoc = GetDocument();
ASSERT_VALID(pDoc);
if (!pDoc)
return;// TODO: add draw code for native data here
// 從文檔中得到數(shù)據(jù)
CArray& m_Array = pDoc->GetBubbleArray(); // 顯示數(shù)據(jù)
for(int nIndex = 0; nIndex < m_Array.GetSize(); ++nIndex )
{
CBubble tempBubble = m_Array.GetAt( nIndex );
pDC->Ellipse(tempBubble.m_nCenterPoint.x - tempBubble.m_fR,
tempBubble.m_nCenterPoint.y - tempBubble.m_fR,
tempBubble.m_nCenterPoint.x + tempBubble.m_fR,
tempBubble.m_nCenterPoint.y + tempBubble.m_fR);
}
}
這樣,我們就實(shí)現(xiàn)了一個(gè)簡(jiǎn)單的支持重啟管理器的文檔視圖類型的MFC應(yīng)用程序。這個(gè)程序可以通過(guò)鼠標(biāo)在視圖中點(diǎn)擊向文檔中添加數(shù)據(jù),然后這些數(shù)據(jù)可以保存和重新打開(kāi)。void CRestartManagerDemoView::OnLButtonDown(UINT nFlags, CPoint point)
{
// TODO: 在此添加消息處理程序代碼和/或調(diào)用默認(rèn)值
CRestartManagerDemoDoc* pDoc = GetDocument();
ASSERT_VALID(pDoc);
if (!pDoc)
return;
CArray& m_Array = pDoc->GetBubbleArray();
// 以當(dāng)前鼠標(biāo)點(diǎn)擊點(diǎn)為圓心,隨機(jī)半徑構(gòu)造一個(gè)CBubble對(duì)象,并添加到文檔中
m_Array.Add( CBubble( point, rand()%30 ));// 更新視圖顯示
Invalidate();CView::OnLButtonDown(nFlags, point);
}然后,我們將這些數(shù)據(jù)在視圖中顯示出來(lái):
void CRestartManagerDemoView::OnDraw(CDC* pDC)
{
CRestartManagerDemoDoc* pDoc = GetDocument();
ASSERT_VALID(pDoc);
if (!pDoc)
return;// TODO: add draw code for native data here
// 從文檔中得到數(shù)據(jù)
CArray& m_Array = pDoc->GetBubbleArray(); // 顯示數(shù)據(jù)
for(int nIndex = 0; nIndex < m_Array.GetSize(); ++nIndex )
{
CBubble tempBubble = m_Array.GetAt( nIndex );
pDC->Ellipse(tempBubble.m_nCenterPoint.x - tempBubble.m_fR,
tempBubble.m_nCenterPoint.y - tempBubble.m_fR,
tempBubble.m_nCenterPoint.x + tempBubble.m_fR,
tempBubble.m_nCenterPoint.y + tempBubble.m_fR);
}
}
這樣,我們就實(shí)現(xiàn)了一個(gè)簡(jiǎn)單的支持重啟管理器的文檔視圖類型的MFC應(yīng)用程序。這個(gè)程序可以通過(guò)鼠標(biāo)在視圖中點(diǎn)擊向文檔中添加數(shù)據(jù),然后這些數(shù)據(jù)可以保存和重新打開(kāi)。
使用Restart Manassas API測(cè)試重啟管理器
接下來(lái),我們可以編寫(xiě)一個(gè)測(cè)試程序,使用Restart Manassas API模擬軟件的更新后重啟,以驗(yàn)證其重啟管理器是否正常工作。
用Visual Studio 2010 CTP創(chuàng)建一個(gè)控制臺(tái)應(yīng)用程序TestRM,然后將其實(shí)現(xiàn)如下:
//#include "stdafx.h"
#include
#includeint _tmain(int argc, _TCHAR* argv[])
{
DWORD dwSessionHandle = 0;
WCHAR wszSessionKey[CCH_RM_SESSION_KEY+1];// 設(shè)定需要重啟的資源
LPCWSTR pwzResourcesToRestart[] =
{L"C:\\Users\\TFSSETUP\\Documents\\Visual Studio 10\\Projects\\
RestartManagerDemo\\Debug\\RestartManagerDemo.exe" };// 創(chuàng)建一個(gè)重啟任務(wù)
if (RmStartSession(&dwSessionHandle, 0, wszSessionKey) == ERROR_SUCCESS)
{
// 注冊(cè)資源
if (RmRegisterResources(dwSessionHandle, 1,
pwzResourcesToRestart, 0, NULL, 0, NULL) == ERROR_SUCCESS)
{
// 關(guān)閉應(yīng)用程序
if (RmShutdown(dwSessionHandle,
RmShutdownOnlyRegistered, NULL) == ERROR_SUCCESS)
{
// 重新啟動(dòng)應(yīng)用程序
if (RmRestart(dwSessionHandle, 0, NULL) == ERROR_SUCCESS)
{
return 0;
}
}
}
}
return 0;
}
我們首先運(yùn)行RestartManagerDemo,在視圖中用鼠標(biāo)點(diǎn)擊向文檔中添加數(shù)據(jù),然后保持文檔為demo.bub。
圖3 支持重啟管理器的MFC應(yīng)用程序
現(xiàn)在,我們就可以運(yùn)行TestRM重啟這個(gè)應(yīng)用程序了。運(yùn)行TestRM后,我們會(huì)看到RestartManagerDemo會(huì)被關(guān)閉然后重新打開(kāi)。同時(shí),我們之前打開(kāi)的文檔demo.bub也被重新加載,整個(gè)應(yīng)用程序很快恢復(fù)到了我們之前的工作狀態(tài)。
【編輯推薦】