ON_COMMAND_RANGE多個按鈕響應(yīng)一個函數(shù)的解決方法
本文描述了ON_COMMAND_RANGE多個按鈕響應(yīng)一個函數(shù)的解決方法。
開發(fā)人員需要注意在自定義消息響應(yīng)函數(shù)的聲明過程中,一定要注意參數(shù)的形式,稍微一疏忽就會導(dǎo)致莫須有的錯誤,具體以O(shè)N_COMMAND_RANGE為例說下。
1.聲明消息響應(yīng)函數(shù):在要添加的工程上添加函數(shù)afx_msg void OnButtonPort();
2.消息映射:
BEGIN_MESSAGE_MAP(CXXXDlg, CDialog) //{{AFX_MSG_MAP(CXXXDlg) ON_WM_SYSCOMMAND() ON_WM_PAINT() ON_WM_TIMER() //}}AFX_MSG_MAP //這里的IDC_BUTTON_PORT_1和 IDC_BUTTON_START_ALL之間有很多個Button,并且ID連續(xù) ON_COMMAND_RANGE(IDC_BUTTON_PORT_1, IDC_BUTTON_START_ALL, OnButtonPort) ON_WM_DEVICECHANGE() END_MESSAGE_MAP()
3.映射函數(shù)的實現(xiàn):實現(xiàn)你自己的響應(yīng)函數(shù) void CXXXDlg::OnButtonPort()
注:此代碼DEBUG OK,Relase異常,不可直接參考,且聽下面分解:
DEBUG通過,不料Release卻直接崩潰,寫了這么多年的CODE還真第一次遇到這種情況,為什么ON_COMMAND_RANGE Debug正常,Release不正常呢?
先MSDN:
Use this macro to map a contiguous range of command IDs to a single message handler function. ON_COMMAND_RANGE(id1, id2, memberFxn ) Parameters id1 Command ID at the beginning of a contiguous range of command IDs. id2 Command ID at the end of a contiguous range of command IDs. memberFxn The name of the message-handler function to which the commands are mapped. Remarks The range of IDs starts with id1 and ends with id2. Use ON_COMMAND_RANGE to map a range of command IDs to one member function. Use ON_COMMAND to map a single command to a member function. Only one message-map entry can match a given command ID. That is, you can't map a command to more than one handler. For more information on mapping message ranges, see Handlers for Message-Map Ranges. There is no automatic support for message map ranges, so you must place the macro yourself.
MSDN也沒有特別說明要注意什么的,我覺得我用的也很正常,于是在網(wǎng)上又搜了一大會,有一個網(wǎng)友非常專業(yè)的解釋的原因,具體網(wǎng)址是:http://yiyunscu.blog.163.com/blog/static/3626332020099802057982/
以下是轉(zhuǎn)載內(nèi)容:
該網(wǎng)友定義如下:
afx_msg void OnCommandMy(WPARAM wParam, LPARAM lParam );
申明只適用于ON_COMMAND消息的函數(shù)申明, 而ON_COMMAND_RANGE的函數(shù)申明在MSDN中建議寫成這樣:
OnCommandMy(UINT nID);
通過switch(nID) case **:進(jìn)行針對不同菜單進(jìn)行消息響應(yīng).
nID就是菜單傳入消息的ID號, 奇怪的是, 在Debug版本下, 先前的申明方式運(yùn)行完全正常, 查閱了MSDN, 找出了可能的原因:
Handler functions for single commands normally take no parameters. With the exception of update handler functions, handler functions for message-map ranges require an extra parameter, nID, of type UINT. This parameter is the first parameter. The extra parameter accommodates the extra command ID needed to specify which command the user actually chose.
針對單個Command消息響應(yīng)函數(shù)可以不帶參數(shù), 但是對于多個Command消息如ON_COMMAND_RANGE申明的消息響應(yīng)需要將函數(shù)參數(shù)列表中的第一個參數(shù)定義為UINT nID, 指明command 的ID號, 按照MSDN的理解, ON_COMMAND_RANGE也可以像ON_COMMAND那樣在消息響應(yīng)函數(shù)中定義兩個參數(shù), 如afx_msg void OnCommandMy(WPARAM wParam, LPARAM lParam );在Debug和Release下, 編譯不會出現(xiàn)問題, 在Debug下運(yùn)行也不會出現(xiàn)問題, 但是在Release下面卻出現(xiàn)內(nèi)存錯誤, 所以可以帶多個參數(shù)感覺只能在Debug下可以行的能, 在Release下就沒失效了.
查閱相關(guān)的資料并利用VC查看相應(yīng)的匯編代碼發(fā)現(xiàn), 應(yīng)該是函數(shù)調(diào)用和返回時棧操作不平衡導(dǎo)致Release版本下出現(xiàn)了內(nèi)存錯誤的問題, ON_COMMAND_RANGE在MFC默認(rèn)的消息響應(yīng)函數(shù)中, 參數(shù)只有一個, 如:
#define ON_COMMAND_RANGE(id, idLast, memberFxn) \ { WM_COMMAND, CN_COMMAND, (WORD)id, (WORD)idLast, AfxSig_vw, \ (AFX_PMSG)(void (AFX_MSG_CALL CCmdTarget::*)(UINT))&memberFxn }, // ON_COMMAND_RANGE(id, idLast, OnFoo) is the same as // ON_CONTROL_RANGE(0, id, idLast, OnFoo)
函數(shù)調(diào)用過程中, 會將傳入的參數(shù)進(jìn)行壓棧操作, 因為MFC默認(rèn)的傳入?yún)?shù)只有一個, 因此調(diào)用OnCommandMy時會有系統(tǒng)傳入的一個消息參數(shù)進(jìn)行壓棧操作. 在函數(shù)返回時, 應(yīng)該進(jìn)行出棧操作, 并且保證調(diào)用完成后棧維持平衡, 否則會出現(xiàn)可能的內(nèi)存錯誤.在DEBUG上沒有出現(xiàn)內(nèi)存錯誤在于在調(diào)用OnCommandMy函數(shù)返回時編譯器在返回代碼處添加了如下的匯編代碼:
pop edi pop esi pop ebx add esp, 48h cmp ebp, esp call __chkesp (0041e680) mov esp, ebp pop ebp ret 8(兩個參數(shù)出棧)
此匯編代碼的作用就是在函數(shù)返回時檢查調(diào)用中和調(diào)用返回時的棧是否一致, 如果不一致, 就強(qiáng)制平棧操作, 因為在這個調(diào)用過程中, 傳入OnCommandMy的消息參數(shù)只有一個(只是申明成兩個, 實際只有一個參數(shù)傳入), 所以存在棧不一致的情況, 但是強(qiáng)制平??梢员苊庥纱艘鸬腻e誤.
在Release版本下, 就沒有了檢測棧的操作,
只是簡單的下面幾句匯編代碼完成出棧操作:
mov esp, ebp pop ebp ret 8兩個參數(shù)出棧)
可以明顯看到, Release下出現(xiàn)了棧操作不平衡的情況, 即入棧數(shù)小于出棧數(shù), 從而導(dǎo)致棧區(qū)地址錯誤, 當(dāng)其它函數(shù)兩次對棧區(qū)進(jìn)行地址訪問時就極有可能出現(xiàn)內(nèi)存錯誤的現(xiàn)象了.
所以, 平時寫程序時在Debug下高度完成之后, 最好還在Release下看一下, 因為有些時候, Debug下對函數(shù)參數(shù)的檢查不是那么嚴(yán)格, 并且在棧的操作上, Debug可以幫助我們解決很多隱藏的問題, 但是Release下就不會了. 另外在自定義的消息響應(yīng)函數(shù)中, Debug和Release都不會對響應(yīng)函數(shù)的參數(shù)列表與MFC默認(rèn)參數(shù)列表進(jìn)行一致性檢測, 從而可能隱藏重大的內(nèi)存出錯的可能性, 導(dǎo)致最終軟件在Release下運(yùn)行可能發(fā)生崩潰.
終于明白了,原來是ON_COMMAND_RANGE只能帶一個參數(shù),帶兩個或不帶都會異常所以重新定義:
afx_msg void OnButtonPort(UINT nID);
而且此nID就是你點擊的按鈕ID值,再也不用之前的麻煩代碼了
CWnd *pWnd = GetFocus(); int nPortID = pWnd->GetDlgCtrlID() ;
問題解決!
附加:
1、ON_COMMAND(ID_VIEW_CUSTOMIZE, OnViewCustomize)==>void CMainFrame::OnViewCustomize();或void CMainFrame::OnViewCustomize(WPARAM wParam, LPARAM lParam);
2、ON_REGISTERED_MESSAGE(AFX_WM_RESETTOOLBAR, OnToolbarReset)==>afx_msg LRESULT CMainFrame::OnToolbarReset(WPARAM /*wp*/,LPARAM);
3、ON_COMMAND_RANGE(ID_SHORTCUT_1, ID_SHORTCUT_5, OnOutlookBarShortcut)==>void CMainFrame::OnOutlookBarShortcut(UINT id);
4、ON_UPDATE_COMMAND_UI(ID_VIEW_CAPTIONBAR, OnUpdateViewCaptionBar)==>void CMainFrame::OnUpdateViewCaptionBar(CCmdUI* pCmdUI);
欄 目:C語言
下一篇:C++實現(xiàn)讀取特定路徑下文件夾及文件名的方法
本文標(biāo)題:ON_COMMAND_RANGE多個按鈕響應(yīng)一個函數(shù)的解決方法
本文地址:http://mengdiqiu.com.cn/a1/Cyuyan/3600.html
您可能感興趣的文章
- 01-10wince程序防止創(chuàng)建多個實例實現(xiàn)互斥作用
- 01-10Objective-C中常用的結(jié)構(gòu)體NSRange,NSPoint,NSSize(CGSize),NSRect實例分析
- 01-10詳解設(shè)計模式中的Command命令模式及相關(guān)C++實現(xiàn)
- 01-10C#使用反射加載多個程序集的實現(xiàn)方法
- 01-10C++11的for循環(huán),以及范圍Range類的簡單實現(xiàn)
- 01-10C++中可以接受任意多個參數(shù)的函數(shù)定義方法(詳解)
- 01-10C/C++中一次性執(zhí)行多個DOS命令的實現(xiàn)思路


閱讀排行
本欄相關(guān)
- 04-02c語言函數(shù)調(diào)用后清空內(nèi)存 c語言調(diào)用
- 04-02func函數(shù)+在C語言 func函數(shù)在c語言中
- 04-02c語言的正則匹配函數(shù) c語言正則表達(dá)
- 04-02c語言用函數(shù)寫分段 用c語言表示分段
- 04-02c語言中對數(shù)函數(shù)的表達(dá)式 c語言中對
- 04-02c語言編寫函數(shù)冒泡排序 c語言冒泡排
- 04-02c語言沒有round函數(shù) round c語言
- 04-02c語言分段函數(shù)怎么求 用c語言求分段
- 04-02C語言中怎么打出三角函數(shù) c語言中怎
- 04-02c語言調(diào)用函數(shù)求fibo C語言調(diào)用函數(shù)求
隨機(jī)閱讀
- 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文
- 01-10SublimeText編譯C開發(fā)環(huán)境設(shè)置
- 01-10delphi制作wav文件的方法
- 08-05DEDE織夢data目錄下的sessions文件夾有什
- 01-10C#中split用法實例總結(jié)
- 08-05dedecms(織夢)副欄目數(shù)量限制代碼修改
- 08-05織夢dedecms什么時候用欄目交叉功能?
- 04-02jquery與jsp,用jquery
- 01-10使用C語言求解撲克牌的順子及n個骰子
- 01-11ajax實現(xiàn)頁面的局部加載