1. 程式人生 > >MFC建立空文件失敗分析

MFC建立空文件失敗分析

轉載自checky_1981:

關於"建立空文件失敗"的問題的分析! 許多新手在遇到此類問題時總是措手無策,如果誰有耐心就看看我寫的下面這片文章吧。 這類問題的出現主要在BOOL CWinApp::ProcessShellCommand(CCommandLineInfo& rCmdInfo); 函式的關鍵內容: BOOL bResult = TRUE; switch (rCmdInfo.m_nShellCommand) { case CCommandLineInfo::FileNew: // 新建 if (!AfxGetApp()->OnCmdMsg(ID_FILE_NEW, 0, NULL, NULL)) OnFileNew(); if (m_pMainWnd == NULL) bResult = FALSE; break; case CCommandLineInfo::FileOpen: if (!OpenDocumentFile(rCmdInfo.m_strFileName)) bResult = FALSE; break; 通過上面的內容我們可以看出:如果沒有對ID_FILE_NEW做對映的話出現問題就在OnFileNew(); CWinApp對OnFileNew的預設實現是呼叫m_pDocManager->OnFileNew(); 我們繼續解析CDocManager,它究竟幹了些什麼? (首先說明一下CDocManager它主要的功能是幫助CWinApp是管理文件模板連結串列和註冊檔案型別.) //如果模板列表為空的話 if (m_templateList.IsEmpty()) { TRACE0("Error: no document templates registered with CWinApp.\n"); AfxMessageBox(AFX_IDP_FAILED_TO_CREATE_DOC);  //報錯並返回.這裡不會報建立新文件出錯。 return; } CDocTemplate* pTemplate = (CDocTemplate*)m_templateList.GetHead(); if (m_templateList.GetCount() > 1) { // more than one document template to choose from // bring up dialog prompting user CNewTypeDlg dlg(&m_templateList); int nID = dlg.DoModal(); if (nID == IDOK) pTemplate = dlg.m_pSelectedTemplate; else return;     // none - cancel operation } ASSERT(pTemplate != NULL); ASSERT_KINDOF(CDocTemplate, pTemplate); pTemplate->OpenDocumentFile(NULL); 通過上面的程式碼我們可以看出,CWinApp的OnFileNew和OnFileOpen分別呼叫CDocManager的虛擬函式OnFileNew 和OnFileOpen。而在CDocManager裡面。通過模板連結串列選擇不同的模板來呼叫文件模板的OpenDocumentFile(); 如果傳入引數NULL表示新建檔案。 下面我們來看看CDocTemplate::OpenDocumentFile()它是一個最關鍵的函式。因為他是虛擬函式,我們考慮 CSingleDocTemplate::OpenDocumentFile的情況。 這個函式裡面有一段程式碼: 其中:AFX_IDP_FAILED_TO_CREATE_DOC 就是字元“建立空文件失敗”的資源id // create a new document pDocument = CreateNewDocument(); ASSERT(pFrame == NULL);     // will be created below bCreated = TRUE; if (pDocument == NULL) { AfxMessageBox(AFX_IDP_FAILED_TO_CREATE_DOC); return NULL; } ASSERT(pDocument == m_pOnlyDoc); if (pFrame == NULL) { ASSERT(bCreated); // create frame - set as main document frame BOOL bAutoDelete = pDocument->m_bAutoDelete; pDocument->m_bAutoDelete = FALSE; // don't destroy if something goes wrong pFrame = CreateNewFrame(pDocument, NULL); pDocument->m_bAutoDelete = bAutoDelete; if (pFrame == NULL) { AfxMessageBox(AFX_IDP_FAILED_TO_CREATE_DOC); delete pDocument;       // explicit delete on error return NULL; } 通過觀察上面的程式碼我們很容易的看出 有兩個可能出錯的原因:1 CreateNewDocument返回為NULL 2  CreateNewFrame 返回為空。 先看 CreateNewDocument() 一般來說這個函式很少失敗。不過在除錯時也不能掉以輕心。 再看看CreateNewFrame()  裡面有一個函式LoadFrame是造成這種“建立新文件失敗”錯誤的源泉所在。 只要它返回False就會彈出這樣的提示。 我們在來看看LoadFrame() 裡面呼叫CFrameWnd::Create()來建立視窗,建立視窗失敗返回Fasle。 這樣問題就變的比較簡單了。 看看Create和CreateEx函式的動作就知道怎麼回事了。 **************************************************************** 1 如果找不到選單資源 返回False 同時也彈出“建立空文件失敗” HINSTANCE hInst = AfxFindResourceHandle(lpszMenuName, RT_MENU); if ((hMenu = ::LoadMenu(hInst, lpszMenuName)) == NULL) { TRACE0("Warning: failed to load menu for CFrameWnd.\n"); PostNcDestroy();            // perhaps delete the C++ object return FALSE; } 2 過載了PreCreateWindow而且返回False也會導致彈出“建立空文件失敗” 3 在OnCreate 裡面返回-1 也會導致彈出“建立空文件失敗”。 ****************************************************************** 以上就是我分析的 出現這樣“建立空文件失敗”問題的大致原因。也許還有其他的原因。我這裡就不一一列舉 了。 說實話遇到此類問題,如果是新手的話,是很難解決的。我寫這篇文件就當是拋磚引玉吧。 我的Email: 

[email protected]