• <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>

            無我

            讓內(nèi)心永遠燃燒著偉大的光明的精神之火!
            靈活的思考,嚴謹?shù)膶崿F(xiàn)
            豪邁的氣魄、頑強的意志和周全的思考

            截獲全局鼠標消息研究

                        想做一個工具,當鼠標移動時即時查看鼠標當前的坐標,現(xiàn)在終于解決,過程記錄如下:

                        首先,為了要捕獲鼠標移動消息并獲取其坐標,我首先想到的是鉤子,于是采用SetWindowsHookEx函數(shù),利用本函數(shù)給整個系統(tǒng)加載鉤子能實現(xiàn)本功能,不過對系統(tǒng)消耗較大,用法可以參見SetWindowsHookEx

                        SetWindowsHookEx使用的關(guān)鍵是以下幾點:

                        1、如果只是鉤特定的某個線程,可以直接在工具代碼中調(diào)用SetWindowsHookEx函數(shù),最后一個參數(shù)填寫要鉤的線程id,但是要鉤整個系統(tǒng)時,最后一個參數(shù)必須填0,并且SetWindowsHookEx和鉤子回調(diào)函數(shù)都必須在dll中實現(xiàn),第三個參數(shù)填寫dll的實例句柄。測試發(fā)現(xiàn):如果建立全局鉤子,第四個參數(shù)填0,但是實現(xiàn)就在本進程中,結(jié)果是:鉤子SetHook正確,返回正確值;如果hook的是鼠標,那么當鼠標一直在本進程窗口上移動時,回調(diào)函數(shù)調(diào)用正確,證明鉤子已經(jīng)在系統(tǒng)中正確安裝并且起作用,如果這些操作后就卸載鉤子UnhookWindowsHookEx,返回值正確;但是如果安裝后,鼠標移出本進程窗口,不管以后是否鼠標再回到進程窗口內(nèi),回調(diào)函數(shù)中的指令都不在執(zhí)行,證明鉤子已經(jīng)壞掉。此時調(diào)用UnhookWindowsHookEx函數(shù),也會發(fā)現(xiàn)返回失敗。錯誤碼1404(1404 用戶清除和刪除失敗 ),進一步說明鉤子已經(jīng)被破壞。原因是鼠標移出窗口進入別的進程,從而要求別的進程加載本鉤子模塊,但是這是訪問別的進程的地址空間,所以失敗。可能windows在檢測到這個錯誤時就會自動清理掉這個失敗的鉤子。所以鉤子回調(diào)函數(shù)失效并且UnhookWindowsHookEx返回失敗。

                        2、運行機制:在Win16環(huán)境中,DLL的全局數(shù)據(jù)對每個載入它的進程來說都是相同的;而在Win32環(huán)境中,DLL函數(shù)中的代碼所創(chuàng)建的任何對象(包括變量)都歸調(diào)用它的線程或進程所有。當進程在載入DLL時,操作系統(tǒng)自動把DLL地址映射到該進程的私有空間,也就是進程的虛擬地址空間,而且也復制該DLL的全局數(shù)據(jù)的一份拷貝到該進程空間。也就是說每個進程所擁有的相同的DLL的全局數(shù)據(jù),它們的名稱相同,但其值卻并不一定是相同的,而且是互不干涉的。
                       在本問題環(huán)境下我們需要想在多個進程中共享數(shù)據(jù),在Win32環(huán)境下就必須進行必要的設置。在訪問同一個Dll的各進程 之間共享存儲器是通過存儲器映射文件技術(shù)實現(xiàn)的。也可以把這些需要共享的數(shù)據(jù)分離出來,放置在一個獨立的數(shù)據(jù)段里,并把該段的屬性設置為共享。必須給這些 變量賦初值,否則編譯器會把沒有賦初始值的變量放在一個叫未被初始化的數(shù)據(jù)段中。方法如下:
                        #pragma data_seg預處理指令用于設置共享數(shù)據(jù)段。例如:
                        #pragma data_seg("SharedDataName")

                        HHOOK hHook = NULL;

                        //其他共享數(shù)據(jù)

                        #pragma data_seg()

                 在#pragma data_seg("SharedDataName")和#pragma data_seg()之間的所有變量將被訪問該Dll的所有進程看到和共享。再加上一條指令#pragma comment(linker,"/section:SharedDataName,rws"),那么這個數(shù)據(jù)節(jié)中的數(shù)據(jù)可以在所有DLL的實例之間共 享。所有對這些數(shù)據(jù)的操作都針對同一個實例的,而不是在每個進程的地址空間中都有一份。當進程隱式或顯式調(diào)用一個動態(tài)庫里的函數(shù)時,系統(tǒng)都要把這個動態(tài)庫映射到這個進程的虛擬地址空間里(以下簡稱"地址空間")。這使得DLL成為進程的一部分,以這個進程的身份執(zhí)行,使用這個進程的堆棧。

                        3、用SetWindowsHookEx方法建立全局鉤子時,有一些顯而易見的缺點:首先值得我們注意的是,Windows鉤子將會降低整個系統(tǒng)的性能,因為它額外增加了系統(tǒng)在消息處理方面的時間;其次,只有當目標進程準備接受某種消息時,鉤子所在的DLL才會被系統(tǒng)映射到該進程的地址空間中,鉤子才能真正開始發(fā)揮作用,因此如果我們要對某些進程的整個生命周期內(nèi)的API調(diào)用情況進行監(jiān)控,這種方法顯然會遺漏某些API的調(diào)用 。

                        以上是用hook實現(xiàn)檢測鼠標方法,顯然hook功能不僅僅這些。不過可能比較消耗系統(tǒng)資源。另外我還發(fā)現(xiàn)一種筆記簡單的方法:利用SetCapture捕獲光標(函數(shù)請參見SetCapture)。該方法很簡單并且有效,只需要在希望捕獲前調(diào)用SetCapture,不要時再調(diào)用ReleaseCapture即可。不過有以下幾點需要注意:

                        1、當進程的窗口調(diào)用的SetCapture以后,他將鎖定所有鼠標輸入,這可能導致窗口的其他按鈕包括關(guān)閉都無法點擊,所以最好在消息循環(huán)中處理某個事件來管理Capture的邏輯,比如鼠標中鍵,鍵盤某個快捷鍵等等。這在調(diào)用ReleaseCapture釋放光標以后都會正常。

                        2、對光標的捕獲如果是超出了當前進程的窗口,是需要先在當前進程窗口下按住鼠標左鍵在移出窗口才有效的!這也是許多spy的處理原理。當時我沒發(fā)現(xiàn)這一點,做了很久總是不能實現(xiàn),很是不解,郁悶n久,白白浪費了很多腦細胞~特此著重提醒各位過往客官!

                        3、用SetCapture注意的問題:設定當前窗口捕獲光標后,如果焦點轉(zhuǎn)移,比如激活了別的窗口,此時本窗口的光標捕獲也不存在了。所以如果有捕獲狀態(tài)的顯示的話,一定要記得處理CaptureChanged消息!

             

                        好了,鼠標捕獲就研究到這里了。聲明:在寫作的過程中,為了在某些點上表達得更清晰,參考了網(wǎng)上一些朋友的博客,有些引用原話,沒有列舉出處,請見諒。有不對的地方,歡迎指正~

            posted on 2009-08-03 17:18 Tim 閱讀(5517) 評論(6)  編輯 收藏 引用 所屬分類: windows系統(tǒng)

            評論

            # re: 截獲全局鼠標消息研究 2009-08-03 18:02 Pear

            只是記錄的話,用WH_JOURNALRECORD吧,不需要DLL  回復  更多評論   

            # re: 截獲全局鼠標消息研究[未登錄] 2009-08-06 09:51 foxriver

            我昨晚還在看這個呢。WH_MOUSE_LL也不需要DLL,而WH_MOUSE就需要了,user32.dll里會做進程DLL注入操作。兩者實現(xiàn)原理不一樣。  回復  更多評論   

            # re: 截獲全局鼠標消息研究 2009-08-06 14:37 Tim

            WH_MOUSE_LL也不需要DLL,而WH_MOUSE就需要,user32.dll里會做進程DLL注入操作。
            ----你是說截獲全局的WH_MOUSE_LL消息不需要將hook寫到dll中嗎?好像不是的呀?能給我看看示例代碼不?謝謝~@foxriver
              回復  更多評論   

            # re: 截獲全局鼠標消息研究 2009-08-10 12:44 虛擬主機

            虛擬主機  回復  更多評論   

            # re: 截獲全局鼠標消息研究 2009-08-11 09:39 Tim

            莫非閣下意思是“到此一游”!~~~@虛擬主機
              回復  更多評論   

            # re: 截獲全局鼠標消息研究 2011-07-05 07:43 游客

            GetRawInputData 會不會效率更高  回復  更多評論   

            <2012年7月>
            24252627282930
            1234567
            891011121314
            15161718192021
            22232425262728
            2930311234

            導航

            統(tǒng)計

            公告

            本博客原創(chuàng)文章,歡迎轉(zhuǎn)載和交流。不過請注明以下信息:
            作者:TimWu
            郵箱:timfly@yeah.net
            來源:www.shnenglu.com/Tim
            感謝您對我的支持!

            留言簿(9)

            隨筆分類(173)

            IT

            Life

            搜索

            積分與排名

            最新隨筆

            最新評論

            閱讀排行榜

            一本久道久久综合狠狠爱| 99精品国产综合久久久久五月天| 亚洲欧美日韩精品久久亚洲区 | 久久久午夜精品| 狠狠色噜噜色狠狠狠综合久久| 精品多毛少妇人妻AV免费久久| 精品国产乱码久久久久久郑州公司| 久久久久久精品免费看SSS| 久久精品aⅴ无码中文字字幕不卡| WWW婷婷AV久久久影片| 亚洲精品久久久www| 久久香蕉超碰97国产精品| 欧美午夜A∨大片久久 | 久久AAAA片一区二区| 狠狠色婷婷久久综合频道日韩| 国产激情久久久久影院老熟女| 久久人人爽人人爽人人片AV东京热| 久久久久国产| 久久96国产精品久久久| 狠狠色丁香久久婷婷综| 亚洲va久久久噜噜噜久久| 久久这里都是精品| 国产精品久久久久久五月尺| 亚洲精品99久久久久中文字幕| 久久AV高潮AV无码AV| 2021国内久久精品| 久久久久亚洲av无码专区导航| 亚洲精品第一综合99久久| 亚洲精品乱码久久久久久按摩 | 日批日出水久久亚洲精品tv| 狠狠色综合网站久久久久久久高清| 99蜜桃臀久久久欧美精品网站 | 亚洲国产精品一区二区久久| 热99re久久国超精品首页| 国产成人香蕉久久久久| 无码久久精品国产亚洲Av影片 | 国产精品免费福利久久| 91精品国产高清久久久久久91| 日本亚洲色大成网站WWW久久| 久久99国产精一区二区三区| 精品国产热久久久福利|