• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            春暖花開
            雪化了,花開了,春天來了
            posts - 149,comments - 125,trackbacks - 0

                  過年前,從china-pub買的。一直也沒有安下心來讀。看今年找工作的境況,也不得不抓把緊了。也愿與c++博客的各位朋友分享我的學習心得。
                   步入主題。

                     這一章開篇介紹了windows函數的幾種返回值:VOID,BOOL,HANDLE,PVOID,LONG/DWORD。讓我們明白,僅僅通過返回值,我們是不能清楚函數調用為什么會失敗的。

                     windows內部,函數檢測到錯誤會采用什么機制呢?它是采用“線程本地存儲區”的機制來講相應的錯誤代碼與“主調線程”關聯到一起。它可以使不同的線程能獨立運行,不會出現相互干擾對方的錯誤代碼的情況。

                     函數返回的時候,其返回值會指出已發生的一個錯誤。

                     我們查看具體是什么錯誤,在相應的函數執行完成后調用GetLastError()即可。

                     windows中,錯誤有三種表示:
                     一個消息ID(如ERROR_PATH_NOT_FOUND)
                        消息文本(如the system cannot find the path specified)
                     一個編號(盡量避免使用)

                     調試程序的時候,我們可以配置watch窗口,讓它始終顯示線程的上一個錯誤代碼和錯誤的文本描述。如$err,hr。hr是要顯示錯誤代碼的消息文本。不過我在windows mobile的環境下沒有成功,沒有弄清楚為什么。

                     那么我們怎么在自己的程序中顯示消息文本呢?文章介紹了利用FormatMessage函數。這里我也介紹一下這個函數的用法:
                     (下面的介紹摘自:http://www.shnenglu.com/bidepan2023/archive/2008/02/03/42433.html
                     DWORD FormatMessage(
                         DWORD dwFlags,
                         LPCVOID lpSource,
                         DWORD dwMessageId,
                         DWORD dwLanguageId,
                         LPTSTR lpBuffer,
                         DWORD nSize,
                         va_list* Arguments
                         );

                     dwFlags:
                     # FORMAT_MESSAGE_ALLOCATE_BUFFER // 此函數會分配內存以包含描述字串。
                     # FORMAT_MESSAGE_FROM_SYSTEM,  // 在系統的id映射表中尋找描述字串
                     # FORMAT_MESSAGE_FROM_HMODULE  // 在其他資源模塊中尋找描述字串
                     # FORMAT_MESSAGE_FROM_STRING   // 消息ID是個字串,不是個DWORD
                     #FORMAT_MESSAGE_IGNORE_INSERTS // 允許我們獲得含有%占位符的消息,不傳遞這個標志,就必須在Arguments參數中提供這些占位符的信息
                     通常為:FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM

                     lpSource:
                     # 指定了FORMAT_MESSAGE_FROM_HMODULE的話,此參數表示模塊的HANDLE
                     # 指定了FORMAT_MESSAGE_FROM_STRING的話,此參數表示id字串
                     通常為:NULL

                     dwMessageId:
                     消息ID;如果指定FORMAT_MESSAGE_FROM_STRING,將被忽略。

                     dwLanguageId:
                     消息描述所用的語言
                     通常為:0表示自動選擇

                     lpBuffer:
                     #如果未指定FORMAT_MESSAGE_ALLOCATE_BUFFER,則為自己提供的緩沖區
                     #否則為系統LocalAlloc分配,需要被用戶LocalFree

                     nSize:
                     #如果未指定FORMAT_MESSAGE_ALLOCATE_BUFFER,則為自己提供的緩沖區大小
                     #否則為系統LocalAlloc分配之最小緩沖區大小

                     Arguments:
                     通常不使用


            例子:

            void ShowError()
            {
                DWORD dwError 
            = GetLastError();

                HLOCAL hlocal 
            = NULL;

                
            // Use the default system locale since we look for Windows messages.
                
            // Note: this MAKELANGID combination has 0 as value
                DWORD systemLocale = MAKELANGID(LANG_NEUTRAL, SUBLANG_NEUTRAL);

                BOOL fOk 
            = FormatMessage(
                    FORMAT_MESSAGE_FROM_SYSTEM 
            | FORMAT_MESSAGE_IGNORE_INSERTS |
                    FORMAT_MESSAGE_ALLOCATE_BUFFER, 
                    NULL, dwError, systemLocale, 
                    (PTSTR) 
            &hlocal, 0, NULL);

                
            if (!fOk) {
                    
            // Is it a network-related error?
                    HMODULE hDll = LoadLibraryEx(TEXT("netmsg.dll"), NULL, 
                        DONT_RESOLVE_DLL_REFERENCES);

                    
            if (hDll != NULL) {
                        fOk 
            = FormatMessage(
                            FORMAT_MESSAGE_FROM_HMODULE 
            | FORMAT_MESSAGE_IGNORE_INSERTS |
                            FORMAT_MESSAGE_ALLOCATE_BUFFER,
                            hDll, dwError, systemLocale,
                            (PTSTR) 
            &hlocal, 0, NULL);
                        FreeLibrary(hDll);
                    }

                }


                
            if (fOk && (hlocal != NULL))
                
            {
                    OutputDebugString((PCTSTR) LocalLock(hlocal));
                    LocalFree(hlocal);
                }

            }



                  這個是書中的例子的代碼,我只是將它歸結為了一個函數ErrorShow。這樣我們在一個函數的后面調用,直接可以知道錯誤的原因。不過環境我是在smart device 的DEBUG環境下調時的,OutputDebugString會輸出相應的字符串。

                  這個例子中同時展示了FormatMessage的兩種用法。觀察一下第二個參數就明白了。

                  visual studio 也提供了一個查詢錯誤的小工具,為Error Lookup。通過以上的示例,我們就知道其相應的工作原理呢。

                  這本書的源碼的下載地址:http://wintellect.com/Books.aspx 
                  大家如果對windows 編程感興趣的話,不妨下來看看。

            posted on 2009-02-20 00:24 Sandy 閱讀(2325) 評論(4)  編輯 收藏 引用 所屬分類: windows學習

            FeedBack:
            # re: Windows核心編程:第一章錯誤處理學習筆記
            2009-02-23 16:25 | 深邃者
            不錯 我也再看這本書 看到第七章了 可以一起討論 呵呵  回復  更多評論
              
            # re: Windows核心編程:第一章錯誤處理學習筆記
            2009-02-24 09:30 | Sandy
            @深邃者
            呵呵,好.一起研究
              回復  更多評論
              
            # re: Windows核心編程:第一章錯誤處理學習筆記
            2009-03-11 19:46 | 鳳凰羽翼
            您好,我也是看了核心編程,有個小問題,就是輸出的時候為什么要用localLock呢,我不用的話也可以輸出啊。
            另:你也畢業生?  回復  更多評論
              
            # re: Windows核心編程:第一章錯誤處理學習筆記
            2009-03-12 10:19 | Sandy
            @鳳凰羽翼
            這個我也是看例子這么寫的。你這么一說,我還專門查了一下。
            http://www.cic.tsinghua.edu.cn/jdx/book4/dlz.htm
            中提到加鎖與解鎖,它是這樣說的 :
            “不管是可移動對象還是可刪除對象,在它分配后其內存句柄是不變的,它是內存對象的恒定引用。但是,應用程序無法通過內存句柄直接存取內存對象,應用程序要存取內存對象還必須獲得它的近地址,這通過調用LocalLock函數實現。LocalLock函數將局部內存對象暫時固定在局部堆的某一位置,并返回該地址的近地址值,此地址可供應用程序存取內存對象使用,它在應用程序調用 LocalUnlock函數解鎖此內存對象之前有效。”

            它的例子如果這樣用的話,那么在
            FormatMessage中調用系統LocalAlloc分配,需要對內存對象進行加鎖與解鎖。

            呵呵,這只是我的理解而已。

            另:你也畢業生?
            對啊,不過我明年才會畢業呢!  回復  更多評論
              
            久久99国产精一区二区三区| 久久亚洲精品视频| 久久久无码精品亚洲日韩京东传媒 | 久久青青草原精品影院| 精品久久人人妻人人做精品| 久久99这里只有精品国产| 久久婷婷五月综合色高清| 大蕉久久伊人中文字幕| 伊人久久大香线蕉综合影院首页| 狠狠色婷婷综合天天久久丁香| 久久久99精品成人片中文字幕 | 手机看片久久高清国产日韩| 久久婷婷五月综合色奶水99啪| 久久久久亚洲精品天堂久久久久久 | 亚洲欧洲精品成人久久曰影片| 狠狠色丁香婷婷久久综合不卡| 久久综合亚洲色HEZYO社区| 国产精品无码久久四虎| 97久久精品人妻人人搡人人玩| 久久丫忘忧草产品| 一本一本久久a久久精品综合麻豆| 国产午夜久久影院| 久久国产精品成人影院| 香蕉久久夜色精品国产2020 | 国产一区二区三区久久| 久久亚洲中文字幕精品有坂深雪| 久久只有这精品99| 色综合久久天天综线观看| 精品免费久久久久国产一区| 91久久国产视频| 国产午夜精品久久久久九九电影| 99久久成人国产精品免费| av国内精品久久久久影院| 无码久久精品国产亚洲Av影片| 久久精品国产久精国产果冻传媒| 中文字幕久久亚洲一区| 久久久久久久精品妇女99 | 久久99国产亚洲高清观看首页| 久久久精品2019免费观看| 国产成人久久AV免费| 狠狠色婷婷综合天天久久丁香|