• <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>
            隨筆-90  評論-947  文章-0  trackbacks-0
             

            灰常感謝各位達人昨天的熱心回帖,讓我受益匪淺。我仰望夜空,群星點點,就如各位的點睛之語,在無盡的蒼穹閃耀。這讓我深深地意識到,在這里,不僅可以分享成果,也可以分享困惑、分享寂寞。(開場白到此結束~)

            在平常的編程中,我發現很容易遇到這種結構:

            (1號方案)

            BOOL foo()
            {
               
            BOOL bRet = FALSE;

               
            HANDLE hProcess = OpenProcess(...);

               
            if (hProcess != NULL)
                {
                   
            HANDLE hToken = OpenProcessToken(hProcess, ...);

                   
            if (hToken != NULL)
                    {
                       
            // ...

                       
            if (LookupPrivilegeValue(...))
                        {
                           
            if (AdjustTokenPrivileges(hToken, ...))
                            {
                               
            bRet = TRUE;
                            }
                        }

                       
            CloseHandle(hToken);
                    }

                   
            CloseHandle(hProcess);
                }

               
            return bRet;
            }

            如上寫法,容易造成縮進級別不斷增加。為了避免這種情況,可以改成:

            (2號方案)

            BOOL foo()
            {
               
            HANDLE hProcess = OpenProcess(...);

               
            if (hProcess == NULL)
                {
                   
            return FALSE;
                }

               
            HANDLE hToken = OpenProcessToken(hProcess, ...);

               
            if (hToken == NULL)
                {
                   
            CloseHandle(hProcess);

                   
            return FALSE;
                }

               
            // ...

               
            if (!LookupPrivilegeValue(...))
                {
                   
            CloseHandle(hToken);
                   
            CloseHandle(hProcess);

                   
            return FALSE;
                }

               
            if (!AdjustTokenPrivileges(hToken, ...))
                {
                   
            CloseHandle(hToken);
                   
            CloseHandle(hProcess);

                   
            return FALSE;
                }

               
            CloseHandle(hToken);
               
            CloseHandle(hProcess);

               
            return TRUE;
            }

            這樣,又引來了新的問題,每次 return FALSE 時的清理任務比較麻煩,要是每步操作都引進新的 HANDLE 的話,后續的清理工作就變得非常繁重。有人推薦do…while(0)的結構,有人推薦goto。這兩種形式分別是——

            do…while(0)

            (3號方案)

            BOOL foo()
            {
               
            HANDLE hProcess = OpenProcess(...);

               
            if (hProcess == NULL)
                {
                   
            return FALSE;
                }

               
            BOOL bRet = FALSE;

               
            do
               
            {
                   
            HANDLE hToken = OpenProcessToken(hProcess, ...);

                   
            if (hToken == NULL)
                    {
                       
            break;
                    }

                   
            // ...

                   
            BOOL bRetInner = FALSE;

                   
            do
                   
            {
                       
            if (!LookupPrivilegeValue(...))
                        {
                           
            break;
                        }

                       
            if (!AdjustTokenPrivileges(hToken, ...))
                        {
                           
            break;
                        }

                       
            bRetInner = TRUE;

                    }
            while (0);

                   
            CloseHandle(hToken);

                   
            if (!bRetInner)
                    {
                       
            break;
                    }

                   
            bRet = TRUE;

                }
            while (0);

               
            CloseHandle(hProcess);

               
            return bRet;
            }

            這種結構可以避免每次 return FALSE 前的一堆清理工作,但缺點是,有幾個依賴性的 HANDLE,就要嵌套幾層的 do…while(0),有時候也會遇到需要三四層嵌套的情形。

            goto

            (4.1號方案)

            BOOL foo
            ()
            {
               
            BOOL bRet = FALSE;

               
            HANDLE hProcess = OpenProcess(...);

               
            if (hProcess == NULL)
                {
                   
            goto CLEAR;
                }

               
            HANDLE hToken = OpenProcessToken(hProcess, ...);

               
            if (hToken == NULL)
                {
                   
            goto CLEAR;
                }

               
            // ...

               
            if (!LookupPrivilegeValue(...))
                {
                   
            goto CLEAR;
                }

               
            if (!AdjustTokenPrivileges(hToken, ...))
                {
                   
            goto CLEAR;
                }

               
            bRet = TRUE;

            CLEAR:
               
            if (hToken != NULL)
                {
                   
            CloseHandle(hToken);
                }

               
            if (hProcess != NULL)
                {
                   
            CloseHandle(hProcess);
                }

               
            return bRet;
            }
            (4.2號方案)

            BOOL foo
            ()
            {
               
            BOOL bRet = FALSE;

               
            HANDLE hProcess = OpenProcess(...);

               
            if (hProcess == NULL)
                {
                   
            goto ERROR_LEVEL0;
                }

               
            HANDLE hToken = OpenProcessToken(hProcess, ...);

               
            if (hToken == NULL)
                {
                   
            goto ERROR_LEVEL1;
                }

               
            // ...

               
            if (!LookupPrivilegeValue(...))
                {
                   
            goto ERROR_LEVEL2;
                }

               
            if (!AdjustTokenPrivileges(hToken, ...))
                {
                   
            goto ERROR_LEVEL2;
                }

               
            bRet = TRUE;

            ERROR_LEVEL2:
               
            CloseHandle(hToken);
            ERROR_LEVEL1:
               
            CloseHandle(hProcess);
            ERROR_LEVEL0:
               
            return bRet;
            }

            (左邊和右邊哪種好一點。。。?)

            在這種情形下,goto 的方案似乎是完美的。但是 goto 如果遇到 C++,缺點體現出來了。下面這一段,現在是 do…while(0) 結構(只有一層嵌套,這種結構用在這里還算合理):

            BOOL foo()
            {
               
            HRESULT hr = CoInitializeEx(0, COINIT_MULTITHREADED);

               
            while (true)
                {
                   
            if (FAILED(hr))
                    {
                       
            break;
                    }

                   
            hr = CoInitializeSecurity(NULL, -1, NULL, NULL, RPC_C_AUTHN_LEVEL_DEFAULT, RPC_C_IMP_LEVEL_IMPERSONATE, NULL, EOAC_NONE, NULL);

                   
            if (FAILED(hr))
                    {
                       
            break;
                    }

                   
            CComPtr<IWbemLocator> pLoc = NULL;
                   
            hr = pLoc.CoCreateInstance(CLSID_WbemLocator, NULL, CLSCTX_INPROC_SERVER);

                   
            if (FAILED(hr))
                    {
                       
            break;
                    }

                   
            CComPtr<IWbemServices> pSvc = NULL;
                   
            hr = pLoc->ConnectServer(_T("ROOT\\CIMV2"), NULL, NULL, NULL, 0, NULL, NULL, &pSvc);

                   
            if (FAILED(hr))
                    {
                       
            break;
                    }

                   
            hr = CoSetProxyBlanket(pSvc, RPC_C_AUTHN_WINNT, RPC_C_AUTHZ_NONE, NULL, RPC_C_AUTHN_LEVEL_CALL, RPC_C_IMP_LEVEL_IMPERSONATE, NULL, EOAC_NONE);

                   
            if (FAILED(hr))
                    {
                       
            break;
                    }

                   
            CComPtr<IEnumWbemClassObject> pEnum = NULL;
                   
            _bstr_t bstrLang = _T("WQL");
                   
            _bstr_t bstrSql = _T("SELECT * FROM __InstanceCreationEvent WITHIN 10")
                       
            _T("WHERE TargetInstance ISA 'Win32_LogonSession' AND (TargetInstance.LogonType = 2 OR TargetInstance.LogonType = 11)");
                   
            hr = pSvc->ExecNotificationQuery(bstrLang, bstrSql, WBEM_FLAG_FORWARD_ONLY | WBEM_FLAG_RETURN_IMMEDIATELY, NULL, &pEnum);

                   
            if (FAILED(hr))
                    {
                       
            break;
                    }

                   
            ULONG uCount = 1;
                   
            CComPtr<IWbemClassObject> pNext = NULL;
                   
            hr = pEnum->Next(WBEM_INFINITE, uCount, &pNext, &uCount);

                   
            if (FAILED(hr))
                    {
                       
            break;
                    }

                   
            // ...

                   
            break;
                }

               
            CoUninitialize();

               
            return SUCCEEDED(hr);
            }

            如果改成 goto,則需要把所有需要對象的定義全放到最前面來,不然 goto 會跳過他們的初始化,編譯不過。但是,所有對象都放到最前面定義,又違反了即用即聲明的規則,而且太多了也容易混淆。

            最后,問題是,如果遇到 C++ 的、多層嵌套的,大家一般如何組織代碼呢?

            謝謝!

            posted @ 2010-03-30 09:55 溪流 閱讀(2780) | 評論 (26)編輯 收藏

            我的觀點可能有點激進,我覺得單件模式啥也不是,純粹是個全局變量的貞潔牌坊而已。全局變量如果有必要,用就用了,何必偽裝;如無必要,就算穿上單件模式的馬甲,到頭來也會搞得一片狼籍——隨處可見GetInstance。

            歡迎討論~

            posted @ 2010-03-29 10:58 溪流 閱讀(4469) | 評論 (63)編輯 收藏

            如題,不知道這樣說是不是清楚了。

            就是說,我們把新的類引入我們自己的工程后,

            如果我們的工程打開了預編譯頭,就需要在.cpp加上#include <stdafx.h>,或者關閉本工程或者那個cpp的預編譯頭選項;如果我們的工程關閉了預編譯頭,就要確保那個.cpp里沒有#include <stdafx.h>。

            這樣感覺很不爽。如果是用別人的成品庫,我會覺得最好不改別人的代碼,于是只好改工程里的選項,麻煩。如果自己寫個類,也想讓用的人(雖然通常是我自己)不要每次遇到這個麻煩。

            有沒有某種方法,在代碼里加上類似:
            #pragma precompileheader(close)
            然后就不用管stdafx.h了,直接把文件加到工程里就好了

            posted @ 2010-03-29 10:30 溪流 閱讀(8832) | 評論 (17)編輯 收藏

            網上提得較多的是 2K/XP 的句柄表,以及句柄分配算法。其中 Win2K 的句柄表在 _EPROCESS + 0x128 處,WinXP 在 _EPROCESS + 0x0c4 處。Vista 和 Win7 找遍了 Internet 沒找到,于是只好下載符號表,裝系統自己找。其實也就 dt _EPROCESS 一下了。Vista 在 _EPROCESS + 0x0dc 處,Win7 在 _EPROCESS + 0x0f4 處。以上均是 32 位系統下的地址。句柄分配算法在 Vista 和 Win7 中都沒有變化,和 XP 一樣(至少我的測試結果是這樣的)。

            小記一筆。明天繼續看 64 位的。

            ==================================================
            WinXP x64: 0x158
            Vista x64: 0x160
            Win7 x64: 0x200

            posted @ 2009-11-17 19:18 溪流 閱讀(781) | 評論 (0)編輯 收藏

            陸陸續續搞了一個多月了,不過其實也就一開始的幾天和最近幾天在好好搞。

            前兩天把 Set、Map 寫完的時候,突然發現我還是完全沒有理解 STL 的迭代器所玩的花樣。其中的類型萃取我看出來了,其余的都沒有。我這里的迭代器是很土的,每個容器自顧自的(盡管很“巧合”有幾個一樣的接口)。

            String 類我還想繼續拓展功能。不過沒想好的就是要不要有 Format 功能:如果沒有,使用上或許偶爾會有一點點不方便(如果也不提供數值和字符串相互轉換的函數的話);如果有,基本上不會去手工解釋 %d、%s 之類的了,那么勢必要用到 sprintf 之類的東西了,那么我的零依賴的設想就落空了。

            MultiSet 和 MultiMap 有點兒傾向于不提供了,真有需求的到時候去 Set<List<T>>、Map<List<T>> 好了。

            文件在此,點擊下載(還沒測試仔細,可能有不少 Bug,甚至可能某些函數有語法錯誤沒測到,這點請諒解)

             

            請各位給點意見~

            posted @ 2009-11-09 22:01 溪流 閱讀(2166) | 評論 (32)編輯 收藏
            僅列出標題
            共18頁: First 10 11 12 13 14 15 16 17 18 
            亚洲国产精品无码久久九九| 99久久国产综合精品麻豆| 久久久精品久久久久特色影视| 中文字幕无码久久久| 国内精品久久久久久中文字幕| 国产成人精品久久亚洲高清不卡| 久久精品成人免费观看97| 国产精品久久婷婷六月丁香| 久久久一本精品99久久精品66| 久久国产成人午夜aⅴ影院| 精品国产乱码久久久久久呢| 97精品伊人久久久大香线蕉| 国内精品伊人久久久影院| 国产欧美一区二区久久| 少妇无套内谢久久久久| 色综合久久久久| 久久天堂AV综合合色蜜桃网| 色诱久久av| 久久亚洲2019中文字幕| 97r久久精品国产99国产精| 国产精品伊人久久伊人电影| 久久久久人妻一区精品色| 精品久久久久久久久免费影院 | 久久99九九国产免费看小说| 人人狠狠综合久久亚洲88| 久久人人爽人人爽人人AV东京热| 欧美精品福利视频一区二区三区久久久精品 | 国产精品久久久久久吹潮| 亚洲精品无码专区久久同性男| 精品久久久久久国产| 精品久久久久久亚洲精品 | 久久精品人人做人人爽电影蜜月| 久久综合五月丁香久久激情| 久久久久97国产精华液好用吗| 麻豆精品久久精品色综合| 69SEX久久精品国产麻豆| 久久精品国产第一区二区三区| 亚洲国产一成人久久精品| 久久精品国产欧美日韩99热| 亚洲&#228;v永久无码精品天堂久久| 91精品国产高清久久久久久io|