青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

道。道。道

安全特性不等于安全的特性

   :: 首頁 :: 聯系 :: 聚合  :: 管理

常用鏈接

搜索

  •  

最新評論

對于熟悉Win API編程的同志們來說,windowsx.h這個頭文件應該不會太陌生吧,這次要講的內容就來自這個windowsx.h頭文件。

經常能在msdn上查到這樣一些函數,明明是個函數,而且模樣長得和一般的api函數也一樣一樣的,可卻叫做macro,為什么呢?留意一下函數使用的requirement,你會發現,它的聲明正是在windowsx.h這個頭文件里。

Windowsx.h包含了這樣一些內容:
宏API,窗口消息分流器,控件API;

所有的這些宏定義,可以使你的程序更加安全,簡潔,結構更清晰,大大提高程序的可讀性;其中窗口消息分流器(message cracker)是我們今天要討論的話題,它可以使我們的API程序變得更簡潔。下面就進入我們的主題:(有關windowsx.h的更多內容,可以參考MS Knowledge Base Article #83456.)

消息分流器是Windows提供的一組宏定義,它的兩個最大的作用,用MS的話來說,就是:

● 安全的數據類型,因為消息分流器完成了大量的類型轉換的工作;
● 使程序向32位windows的轉化更簡單;

當然,使用消息分流器會大大改變程序的面貌,你也可以選擇不使用它。

下面我們就以一個對話框窗口的消息處理過程為例,看看消息分流器到底是怎么運作的。


1.消息分流器的基本使用
先看一個普通的窗口消息處理函數,它可能需要處理一些窗口的初始化,無效客戶區重繪等消息:

LRESULT CALLBACK WndProc (HWND hwnd, UINT msg,
WPARAM wParam, LPARAM lParam)
{
switch(msg)
{
? case WM_CREATE:
? // ...
? return 0;
? ?
? case WM_PAINT:
? // ...
? return 0;
? ? ? ? ?
? case WM_DESTROY:
? //...
? return 0;
}
return DefWindowProc(hwnd, msg, wParam, lParam);
}

而通過使用消息分流器,我們可以把每個case都寫到相應的消息處理函數中,就像下面這樣:

LRESULT CALLBACK WndProc (HWND hwnd, UINT msg,
WPARAM wParam, LPARAM lParam)
{
switch(msg)
{
? case WM_CREATE:
? ? ? return HANDLE_WM_CREATE(hwnd, wParam, lParam, Cls_OnCreate);
? ?
? case WM_PAINT:
? ? ? return HANDLE_WM_PAINT(hwnd, wParam, lParam, Cls_OnPaint);
? ? ? ? ?
? case WM_DESTROY:
? return HANDLE_WM_DESTROY(hwnd, wParam, lParam, Cls_OnDestroy);
}
return DefWindowProc(hwnd, msg, wParam, lParam);
}

這里用到了三個宏定義:HANDLE_WM_CREATE, HANDLE_WM_PAINT, HANDLE_WM_DESTROY;這三個宏定義就是我們的三個消息分流器(別看叫什么分流器,說穿了也不值幾個錢,呵呵),它們在windowsx.h中的定義如下:

#define HANDLE_WM_CREATE(hwnd, wParam, lParam, fn) \
((fn)((hwnd), (LPCREATESTRUCT)(lParam)) ? 0L : (LRESULT)-1L)
#define HANDLE_WM_PAINT(hwnd, wParam, lParam, fn) \
((fn)(hwnd), 0L)
#define HANDLE_WM_DESTROYCLIPBOARD(hwnd, wParam, lParam, fn) \
? ((fn)(hwnd), 0L)

把這三個宏定義替換回去,就變成:

LRESULT CALLBACK WndProc (HWND hwnd, UINT msg,
WPARAM wParam, LPARAM lParam)
{
switch(msg)
{
? case WM_CREATE:
? ? ? return Cls_OnCreate(hwnd, (LPCREATESTRUCT)(lParam) ? 0L : (LRESULT)-1L;
? ? // 如果處理了消息,則Cls_OnCreate應返回TRUE,導致WndProc返回0,否則Cls_OnCreate返回FALSE,導致WndProc返回-1;
? case WM_PAINT:
? ? ? return Cls_OnPaint(hwnd), 0L;
? ? // 逗號表達式;Cls_OnPaint是void類型,這里返回0; ? ?
? case WM_DESTROY:
? return Cls_OnDestroy(hwnd), 0L; ? ? ? ? // 同Cls_OnPaint
} ? ?
return DefWindowProc(hwnd, msg, wParam, lParam);
}

之后我們就可以按照消息分流器的定義編寫相應的消息處理函數了:

BOOL Cls_OnCreate(HWND hwnd, LPCREATESTRUCT lpCreateStruct){…};
void Cls_OnPaint(HWND hwnd){…};
void Cls_OnDestroyClipboard(HWND hwnd){…};

windowsx.h還提供了一個更加簡化的方法:使用HANDLE_MSG宏,這個宏是這樣定義的:

#define HANDLE_MSG(hwnd, message, fn) ? \
? case (message): return HANDLE_##message((hwnd), (wParam), (lParam), (fn))

這個宏要做的就是根據不同的message(##用來連接前后的字符串),把自己“變成”相應的HANDLE_XXXXMESSAGE形式的宏,再通過相應的宏來執行消息處理代碼;
比如實際代碼中寫入:

HANDLE_MSG(hwnd, WM_CREATE, Cls_OnCreate)

則經過轉換就變成:

case (WM_CREATE): return HANDLE_WM_CREATE((hwnd), (wParam), (lParam), (Cls_OnCreate))

這樣,我們就可以直接把程序寫為:
LRESULT CALLBACK WndProc (HWND hwnd, UINT msg,
WPARAM wParam, LPARAM lParam)
{
switch(msg)
{
HANDLE_MSG(hwnd, WM_CREATE, Cls_OnCreate);
HANDLE_MSG(hwnd, WM_PAINT, Cls_OnPaint);
HANDLE_MSG(hwnd, WM_DESTROY, Cls_OnDestroy);
}
return DefWindowProc(hwnd, msg, wParam, lParam);
}

之后直接編寫相應的消息處理過程就可以了。是不是簡潔多了?而且把消息處理封裝到函數里面,就可以使用VS直接跳轉到這個函數,再也不用費勁去找那個case了。要注意的一點是,雖然windowsx.h里包括了所有消息對應的分流器,但它們的參數是宏定義顯式說明的,在編寫消息處理函數時,必須遵循宏定義中的參數類型,否則會導致錯誤;這么多消息分流器,我們每次新寫一個消息處理函數時就得看看是否把參數設置正確了,整個過程繁瑣冗長。好在已經有一個工具叫Message Cracker Wizard,可以幫助我們生成消息分流器和相關的處理過程,具體見:http://www.codeproject.com/win32/msgcrackwizard.asp


2.在對話框中使用消息分流器
在對話框消息處理中,窗口子類化是我們經常使用的手段,這也可以通過消息分流器實現,但是有點小問題 :>
下面是一個使用了windowsx.h消息分流器的對話框及其處理過程:
……
int WINAPI _tWinMain(HINSTANCE hinstExe, HINSTANCE, PTSTR pszCmdLine, int)
{
DialogBoxParam(
? ? ? ? hinstExe, MAKEINTRESOURCE(IDD_PASSTHRU), NULL, (DLGPROC)Dlg_Proc, 0);

? return(0);
}
……

LRESULT CALLBACK Dlg_Proc (HWND hwnd, UINT msg,
WPARAM wParam, LPARAM lParam)
{
switch(msg)
{
HANDLE_MSG(hwnd, WM_INITDIALOG, Cls_OnInitDialog); // 不能直接使用HANDLE_MSG宏
HANDLE_MSG(hwnd, WM_COMMAND, Cls_OnCommand); ? ? // 不能直接使用HANDLE_MSG宏
}

return false;
}

以上程序中直接使用HANDLE_MSG可能導致錯誤;為什么呢?問題出在子類化的消息處理過程的返回值上,msdn中對于對話框消息處理過程的返回值有如下說明:

一般情況下,對話框過程函數應該在處理了消息的情況下返回TRUE,如果沒有處理,則返回FALSE。如果對話框過程返回了FALSE,那么對話框管理器為這條消息準備默認的對話操作。

如果對話框處理了一個需要特定返回值的消息,則對話框的返回值應該被設置為調用SetWindowLong后的返回值,并在返回TRUE之前立即返回這個值。注意你必須立即調用SetWindowLong(這個函數用于調用窗口子類化的過程),這會導致DWL_MSGRESULT值被一個嵌套的對話框消息改寫。返回值為特定值的消息有:
? ? ? WM_CHARTOITEM
? ? ? WM_COMPAREITEM
? ? ? WM_CTLCOLORBTN
? ? ? WM_CTLCOLORDLG
? ? ? WM_CTLCOLOREDIT
? ? ? WM_CTLCOLORLISTBOX
? ? ? WM_CTLCOLORSCROLLBAR
? ? ? WM_CTLCOLORSTATIC
? ? ? WM_INITDIALOG
? ? ? WM_QUERYDRAGICON
? ? ? WM_VKEYTOITEM
看到沒有? 我們的消息WM_INITDIALOG也在其中,對這個消息進行處理的過程不能簡單的返回TRUE表示對消息進行了處理,而是另有其意;它將轉化為:

case (WM_INITDIALOG): return HANDLE_WM_INITDIALOG(hwnd, wParam, lParam, Cls_OnInitDialog);

宏HANDLE_WM_INITDIALOG定義如下:

#define HANDLE_WM_INITDIALOG(hwnd, wParam, lParam, fn) \
(LRESULT)(DWORD)(UINT)(BOOL)(fn)((hwnd), (HWND)(wParam), lParam)

對WM_INITDIALOG的處理,如果返回TRUE,則表示設置鍵盤焦點到對話框的默認控件,否則返回FALSE;這時好像還看不出什么問題,而對于我們的另外一個消息WM_COMMAND,HANDLE_MSG簡單的把它變成:

case (WM_COMMAND): return HANDLE_WM_COMMAND(hwnd, wParam, lParam, Cls_OnCommand);

宏HANDLE_WM_COMMAND定義如下:

#define HANDLE_WM_COMMAND(hwnd, wParam, lParam, fn) \
? ((fn)((hwnd), (int)(LOWORD(wParam)), (HWND)(lParam), (UINT)HIWORD(wParam)), 0L)

問題出來了,我們的Cls_OnCommand由于是個void型的函數,是沒有返回值的,因此windows默認這種消息處理過程必須返回一個0值,而返回0值不就表示我們的消息過程不處理這個消息么?這個矛盾是HANDLE_MSG無法解決的。怎么辦才能使消息過程在處理完WM_COMMAND消息之后正確的返回一個TRUE呢? 答案是使用另一個windowsx.h中的宏:SetDlgMsgResult(hwnd, msg, result)

這個宏定義如下:

#define ? SetDlgMsgResult(hwnd, msg, result) ((
? ? (msg) == WM_CTLCOLORMSGBOX ? ? || \
? ? (msg) == WM_CTLCOLOREDIT ? ? || \
? ? (msg) == WM_CTLCOLORLISTBOX ? || \
? ? (msg) == WM_CTLCOLORBTN ? ? ? || \
? ? (msg) == WM_CTLCOLORDLG ? ? ? || \
? ? (msg) == WM_CTLCOLORSCROLLBAR ? || \
? ? (msg) == WM_CTLCOLORSTATIC ? ? || \
? ? (msg) == WM_COMPAREITEM ? ? ? || \
? ? (msg) == WM_VKEYTOITEM ? ? ? || \
? ? (msg) == WM_CHARTOITEM ? ? ? || \
? ? (msg) == WM_QUERYDRAGICON ? ? || \
? ? (msg) == WM_INITDIALOG ? ? ? ? \
? ) ? (BOOL)(result) : (SetWindowLongPtr((hwnd), DWLP_MSGRESULT, (LPARAM)(LRESULT)(result)), TRUE))

(有沒有注意到,里面多了一個WM_CTLCOLORMSGBOX ? 這個消息是16位WinAPI中的消息,一度被轉換為Win32 API的一個消息;現在在最新的32位API中已經被刪除了;保留它可能考慮到兼容性的問題,這里不做進一步討論)
現在看到了,如果對話框過程處理的消息恰巧為返回特定值中的一個,則如實返回result;不要被前面的BOOL蒙蔽,BOOL在頭文件中的定義實際上是一個int型,一旦需要返回非TRUE或FALSE的其他值,照樣可以;這樣,我們的Cls_OnInitDialog就能夠正確的返回它的BOOL值了,而Cls_OnCommand在處理之后,也可以由后面的逗號表達式正確的返回一個TRUE表示消息已處理。

在《Windows核心編程》一書中,大牛Jeffrey自己定義了一個宏,使SetDlgMsgResult宏的使用更加方便:

#define chHANDLE_DLGMSG(hwnd, message, fn) ? ? ? ? ? \
? case (message): return (SetDlgMsgResult(hwnd, uMsg, ? \
? ? HANDLE_##message((hwnd), (wParam), (lParam), (fn))))

可見這個宏只是簡單的對SetDlgMsgRseult宏進行了封裝。

這樣,我們最終的代碼可以寫成:

LRESULT CALLBACK Dlg_Proc (HWND hwnd, UINT msg,
WPARAM wParam, LPARAM lParam)
{
switch(msg)
{
chHANDLE_DLGMSG(hwnd, WM_INITDIALOG, Cls_OnInitDialog); // 使用大牛的chHANDLE_DLGMSG宏
chHANDLE_DLGMSG(hwnd, WM_COMMAND, Cls_OnCommand); ? ?
}

return false;
}



下面把原來程序整個框架列出來:

LRESULT CALLBACK Dlg_Proc(HWND hwnd, UNIT umsg, WPARAM wparam, LPARAM lparam)
{
? ? switch(msg) ? ?
? ? {
? ? case WM_COMMAND: ? ? ? ? ? // 每個case都被一個message cracker代替,這里使用大牛同志的
? ? ? ? // do something; ? ? // chHANDLE_DLGMSG宏;這個宏負責對消息篩選,處理并返回相應的值
? ? ? ? return true;

? ? case WM_INITDIALOG:
? ? ? ? // do something;
? ? ? ? return xxxx;
}

return false; ? ? // 如果消息不在我們的DlgProc過程中被處理,則告訴調用這個DlgProc的消息,
} ? ? ? ? ? ? ? ? ? ? ? //告訴系統的對話框管理器,這個消息我們不處理,交給你了

對比一下,消息分流器的作用不言自明。

以上只是介紹了消息分流器的部分應用,更多創造性的用法還等你自己在實踐中發掘。

下面列出一些有用的參考資料:

http://support.microsoft.com/default.aspx?scid=kb;en-us;83456 介紹了STRICT宏定義以及windowsx.h
http://www.codeproject.com/win32/msgcrackwizard.asp ? ? 提供message cracker wizard的下載,而且附有源代碼
《windows核心編程》windows系統編程,就跟定大牛了 :> 他在自己的sample中大量使用了message cracker
posted on 2006-11-22 16:53 獨孤九劍 閱讀(500) 評論(0)  編輯 收藏 引用 所屬分類: Win32Visual C++ 8.0
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            亚洲深夜福利网站| 尤物精品国产第一福利三区| 亚洲国产综合视频在线观看| 久久久www成人免费毛片麻豆 | 国产午夜一区二区三区| 午夜一区不卡| 久久精品人人做人人爽电影蜜月| 国产日韩欧美视频| 久久久久国色av免费观看性色| 性做久久久久久久久| 红桃视频国产精品| 欧美激情中文不卡| 欧美日韩精品在线播放| 亚洲影视在线| 欧美在线观看一区二区| 亚洲狠狠婷婷| 亚洲手机成人高清视频| 国产精品日韩一区| 久热综合在线亚洲精品| 免费观看一级特黄欧美大片| 99视频精品| 欧美一区二区精品在线| 亚洲电影中文字幕| 一区二区三区四区五区视频| 国产精品一区二区三区成人| 久久综合伊人77777尤物| 欧美大片va欧美在线播放| 亚洲一区二区三区色| 羞羞漫画18久久大片| 亚洲国产mv| 亚洲一区二区三区免费观看| 黄色成人av| 99亚洲一区二区| 雨宫琴音一区二区在线| 一本久道久久综合中文字幕| 韩国女主播一区| 日韩一本二本av| 亚洲国产精品v| 欧美一区网站| 亚洲天堂av在线免费观看| 香蕉成人啪国产精品视频综合网| 亚洲国产精品久久久| 亚洲视频免费在线观看| 亚洲人成在线观看网站高清| 亚洲欧美国产三级| 亚洲青色在线| 久久激情视频久久| 亚洲欧美成人一区二区在线电影| 久久亚洲一区二区| 久久精品国产精品亚洲精品| 欧美性色视频在线| 亚洲国产日韩在线| 亚洲福利av| 欧美一区二区三区精品| 亚洲欧美日韩国产成人| 欧美日韩精品一区视频| 欧美激情网友自拍| 在线观看日产精品| 久久久最新网址| 久久精品国产亚洲一区二区| 欧美深夜福利| 一区二区日韩精品| 日韩午夜激情电影| 欧美成人精品三级在线观看 | 亚洲黄色视屏| 久久成年人视频| 久久久久久久久久久久久久一区| 国产精品羞羞答答| 亚洲午夜精品一区二区| 99人久久精品视频最新地址| 欧美丰满少妇xxxbbb| 亚洲国产视频一区| 亚洲精品一二三| 欧美激情综合色综合啪啪| 欧美成人高清| 亚洲精品欧美| 欧美激情综合| 99国产精品久久| 亚洲女同在线| 国产欧美日韩亚洲| 欧美在线视屏| 欧美va亚洲va香蕉在线| 亚洲激情视频在线| 欧美日韩日本视频| 亚洲综合第一页| 葵司免费一区二区三区四区五区| 永久免费精品影视网站| 欧美va亚洲va香蕉在线| 日韩网站在线看片你懂的| 亚洲视频日本| 国产一区二区精品| 美日韩丰满少妇在线观看| 亚洲国内欧美| 欧美一级片一区| 激情久久久久久久| 欧美日本中文字幕| 亚洲一区高清| 欧美成人一区二区三区| 亚洲一区在线免费观看| 国产亚洲福利社区一区| 美女精品在线| 中国成人在线视频| 免费亚洲一区二区| 亚洲午夜在线观看| 娇妻被交换粗又大又硬视频欧美| 欧美激情二区三区| 欧美一区二区三区婷婷月色| 亚洲国产日韩美| 欧美一区国产二区| 亚洲免费高清视频| 国产亚洲二区| 欧美日韩精品免费观看视频完整| 性做久久久久久久久| 亚洲欧洲综合| 久久天堂av综合合色| 亚洲性色视频| 亚洲国产一区二区视频| 国产日韩欧美视频在线| 欧美日韩视频一区二区三区| 久久久久国色av免费看影院| 日韩亚洲精品电影| 亚洲大胆女人| 久久蜜臀精品av| 午夜精品一区二区三区电影天堂| 亚洲观看高清完整版在线观看| 国产欧美一级| 国产精品久久久久永久免费观看| 欧美成人午夜激情视频| 久久久久天天天天| 午夜精品久久久久久99热软件| 亚洲免费成人av| 亚洲国产精品999| 免费中文字幕日韩欧美| 久久伊伊香蕉| 久久国产精品久久精品国产 | 在线不卡亚洲| 国产一在线精品一区在线观看| 欧美视频手机在线| 欧美日韩国产综合网| 欧美大片第1页| 欧美寡妇偷汉性猛交| 欧美大成色www永久网站婷| 免费成人av在线| 麻豆精品视频在线| 欧美 日韩 国产 一区| 美女精品自拍一二三四| 老司机亚洲精品| 美日韩精品免费观看视频| 久久久久国产免费免费| 久久裸体艺术| 欧美99久久| 欧美精品二区| 国产精品成人一区二区网站软件| 欧美色图首页| 国产精品久久久久9999高清| 欧美性理论片在线观看片免费| 欧美日韩午夜剧场| 欧美日韩系列| 国产精品亚洲综合| 国产专区欧美精品| 在线日韩av永久免费观看| 亚洲国产欧美久久| 亚洲美女在线视频| 亚洲一区国产一区| 久久精品一区二区三区不卡| 久久久久久久久久久一区| 欧美黑人多人双交| 亚洲人成7777| 亚洲影视中文字幕| 久久久久国产精品一区三寸| 免费短视频成人日韩| 欧美日韩国产精品成人| 国产乱码精品一区二区三区av| 国产一区二区三区在线观看免费| 亚洲承认在线| 亚洲视频在线免费观看| 久久激情综合网| 亚洲黄色av| 亚洲综合精品自拍| 老司机午夜精品视频| 欧美日韩中文在线观看| 黑人一区二区三区四区五区| 亚洲激情亚洲| 欧美有码在线观看视频| 欧美99久久| 亚洲欧美另类久久久精品2019| 麻豆91精品| 国产乱码精品一区二区三区五月婷 | 免费欧美网站| 国产精品久久久久毛片大屁完整版| 国产在线播精品第三| 亚洲狼人精品一区二区三区| 久久精品一区二区三区不卡| 亚洲精品影视| 久久一区二区三区四区| 国产精品香蕉在线观看| 亚洲欧洲在线一区| 久久久久国产精品麻豆ai换脸| 亚洲每日更新| 欧美91大片|