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

            牽著老婆滿街逛

            嚴以律己,寬以待人. 三思而后行.
            GMail/GTalk: yanglinbo#google.com;
            MSN/Email: tx7do#yahoo.com.cn;
            QQ: 3 0 3 3 9 6 9 2 0 .

            金山詞霸”屏幕取詞技術(shù)揭密(討論稿)

            文章來源:http://blog.joycode.com/yaodong/articles/25506.aspx


            這篇文章最早是發(fā)在北極星論壇的一系列帖子,那時候聞怡洋(好像他也是MVP)也在那里混
            原始的帖子我已經(jīng)沒有了,但不知道是誰幫我收集整理了下來(非常感謝),我用google找到了
            ?
            這是我進金山之前寫的,應該不算泄露公司技術(shù)秘密吧 吐舌笑臉
            而且這些現(xiàn)在看來似乎已經(jīng)有些過時了
            ?
            那時討論的只是Win31和Win9x下的取詞實現(xiàn)
            ?
            我到了金山之后不是負責取詞模塊,而是做UI,因為有個家伙比我更擅長做這種東西
            他用SoftIce調(diào)試匯編代碼非常熟練,做逆向工程方面有過人的天分。
            ?
            ?
            “亦東” 是我那時的筆名
            ?
            ?
            “金山詞霸”屏幕取詞技術(shù)揭密(討論稿)
            ?
            主題  屏幕取詞技術(shù)系列講座(一)
            作者   亦東
            很多人對這個問題感興趣。
            原因是這項技術(shù)讓人感覺很神奇,也很有商業(yè)價值。
            現(xiàn)在詞典市場金山詞霸占了絕對優(yōu)勢,所以再做字典也沒什么前途了。我就是這么認為的,所以我雖然掌握了這項技術(shù),卻沒去做字典軟件。只做了一個和詞霸相似的軟件自己用,本來想拿出來做共享軟件,但我的詞庫是“偷”來的,而且詞匯不多,所以也就算了,詞庫太小,只能取詞有什么用呢?而且詞霸有共享版的。
            但既然很多人想了解這項技術(shù),我也不會保留。我準備分多次講述這項技術(shù)的所有細節(jié)。
            大約每周一兩次。想知道的人就常常來看看吧!
            一.基礎(chǔ)知識
            首先想編這種程序需要一些基礎(chǔ)知識。
            會用Vc++,包括16/32位。
            精通Windows API特別是GDI,KERNEL部分。
            懂匯編語言,會用softice調(diào)試程序,因為這種程序最好用softice調(diào)試。
            二.基本原理
            在Window 3.x時代,windows系統(tǒng)提供的字符輸出函數(shù)只有很少的幾個。
            TextOut
            ExtTextOut
            DrawText
            ......
            其中DrawText最終是用ExtTextOut實現(xiàn)的。
            所以Windows的所有字符輸出都是由調(diào)用TextOut和ExtTextOut實現(xiàn)的。因此,如果你可以修改這兩個函數(shù)的入口,讓程序先調(diào)用你自己的一個函數(shù)再調(diào)用系統(tǒng)的字符輸出,你就可以得到Windows所有輸出的字符了。
            到了Windows95時代,原理基本沒變,但是95比3.x要復雜。開始的時候,一些在windows3.x下編寫的取詞軟件仍然可以是使用。但是后來出了個IE4,結(jié)果很多詞典軟件就因為不支持IE4而被淘汰了,但同時也給一些軟件創(chuàng)造了機會,如金山詞霸。其實IE4的問題并不復雜,只不過它的輸出的是unicode字符,是用TextOutW和ExtTextOutW輸出的。知道了這一點,只要也截取就可以了。不過實現(xiàn)方法復雜一點,以后會有詳細講解。現(xiàn)在又出了個IE5,結(jié)果詞霸也不好用了,微軟真是#^@#$%$*&^&#@#@..........
            我研究后找到了一種解決辦法,但還有些問題,有時會取錯,正在繼續(xù)研究,希望大家共同探討。
            另外還有WindowsNT,原理也是一樣,只是實現(xiàn)方法和95下完全不同。
            三.技術(shù)要點
            要實現(xiàn)取詞,主要要解決以下技術(shù)問題。
            1.截取API入口,獲得API的參數(shù)。
            2.安全地潛入Windows內(nèi)部,良好地兼容Windows的各個版本
            3.計算鼠標所在的單詞和字母。
            4.如果你在Window95下,做32位程序,還涉及Windows32/16混合編程的技術(shù)。
            今天先到這里吧!最好準備一份softice for 95/98和金山詞霸,讓我們先來分析一下別人是怎么做的。
            歡迎與我聯(lián)系
            E-Mail:yeedong@163.net
            主題  屏幕取詞技術(shù)系列講座(二)
            作者   亦東
            很抱歉讓大家久等了!
            我看了一些人的回帖,發(fā)現(xiàn)很多人對取詞的原理還是不太清楚。
            首先我來解釋一下hook問題。詞霸中的確用到了hook,而且他用了兩種hook其中一種是Windows標準hook,通過SetWindowHook安裝一個回調(diào)函數(shù),它安裝了一個鼠標hook,是為了可以及時響應鼠標的消息用的和取詞沒太大關(guān)系。
            另一種鉤子是API鉤子,這才是取詞的核心技術(shù)所在。他在TextOut等函數(shù)的開頭寫了一個jmp語句,跳轉(zhuǎn)到自己的代碼里。
            你用softice看不到這個跳轉(zhuǎn)語句是因為它只在取詞的一瞬間才存在,平時是沒有的。
            你可以在TextOut開頭設(shè)一個讀寫斷點
            bpm textout
            再取詞,就會找到詞霸用來寫鉤子的代碼了。
            /**********************************
            所以我在次強調(diào),想學這種技術(shù)一定要懂匯編語言和熟練使用softice.
            **********************************/
            至于從cjktl95中dump出來的未公開函數(shù)是和Windows32/16混合編程有關(guān)的,以后我會提到他們。
            我先來講述取詞的過程,
            0 判斷鼠標是否在一個地方停留了一段時間
            1 取得鼠標當前位置
            2 以鼠標位置為中心生成一個矩形
            3 掛上API鉤子
            4 讓這個矩形產(chǎn)生重畫消息
            5 在鉤子里等輸出字符
            6 計算鼠標在哪個單詞上面,把這個單詞保存下來
            7 如果得到單詞則摘掉API鉤子,在一段時間后,無論是否得到單詞都摘掉API鉤子
            8 用單詞查詞庫,顯示解釋框。
            很多步驟實現(xiàn)起來都有一些難度,所以在中國可以做一個完善的取詞詞典的人屈指可數(shù)。
            其中0,1,2,7,8比較簡單就不提了。
            先說如何掛鉤子:
            所謂鉤子其實就是在WindowsAPI入口寫一個JMP XXXX:XXXX語句,跳轉(zhuǎn)到自己的代碼里。
            步驟如下:
            1.取得Windows API入口,用GetProcAddress實現(xiàn)
            2.保存API入口的前五個字節(jié),因為JMP是0xEA,地址是4個字節(jié)
            3.寫入跳轉(zhuǎn)語句
            這步最復雜
            Windows的代碼段本來是不可以寫的,但是Microsoft給自己留了個后門。
            有一個未公開函數(shù)是AllocCsToDsAlias,
            UINT WINAPI ALLOCCSTODSALIAS(UINT);
            你可以取到這個函數(shù)的入口,把API的代碼段的選擇符(要是不知道什么是選擇符,就先去學學保護模式編程吧)傳給他,他會返回一個可寫的數(shù)據(jù)段選擇符。這個選擇符用完要釋放的。用新選擇符和API入口的偏移量合成一個指針就可以寫windows的代碼段了。
            這就是取詞技術(shù)的最核心的東東,不止取詞,連外掛中文平臺全屏漢化都是使用的這種技術(shù)。現(xiàn)在知道為什么這么簡單的幾句話卻很少知道了吧?因為太多的產(chǎn)品使用他,太多的公司靠他賺錢了。
            這些公司和產(chǎn)品有:中文之星,四通利方,南極星,金山詞霸,實達銘泰的東方快車,roboword,譯典通,即時漢化專家等等等等。。。。還有至少20多家小公司。他們的具體實現(xiàn)雖然不同,但大致原理是相同的。
            我這些都是隨手寫的,也沒有提綱之類的東西,以后如果有機會我會整理一下,大家先湊合著看吧!xixi...
            ?
            主題  關(guān)于屏幕取詞的討論(三)
            作者   亦東

            讓大家久等,很抱歉,前些時候工作忙硬盤又壞了,太不幸了。
            這回來點真格的。
            咱們以截取TextOut為例。
            下面是代碼:
            //截取TextOut?
            typedef?UINT?(WINAPI*?ALLOCCSTODSALIAS)(UINT);?
            ALLOCCSTODSALIAS?AllocCsToDsAlias;?
            BYTE?NewValue[
            5];//保存新的入口代碼?
            BYTE?OldValue[5];//API原來的入口代碼?
            unsigned?char?*?Address=NULL;//可寫的API入口地址?
            UINT?DsSelector=NULL;//指向API入口的可寫的選擇符?
            WORD?OffSetEntry=NULL;//API的偏移量?
            BOOL?bHookAlready?=?FALSE;?//是否掛鉤子的標志?
            BOOL?InitHook()?
            {?
            HMODULE?hKernel,hGdi;?
            hKernel?
            =?GetModuleHandle("Kernel");?
            if(hKernel==NULL)?
            return?FALSE;?
            AllocCsToDsAlias?
            =?(ALLOCCSTODSALIAS)GetProcAddress(hKernel,"AllocCsToDsAlias");//這是未公開的API所以要這樣取地址?
            if(AllocCsToDsAlias==NULL)?
            return?FALSE;?
            hGdi?
            =?GetModuleHandle("Gdi");?
            if(hmGdi==NULL)?
            return?FALSE;?
            FARPROC?Entry?
            =?GetProcAddress(hGdi,"TextOut");?
            if(Entry==NULL)?
            return?FALSE;?
            OffSetEntry?
            =?(WORD)(FP_OFF(Entry));//取得API代碼段的選擇符?
            DsSelector?=?AllocCsToDsAlias(FP_SEG(Entry));//分配一個等同的可寫的選擇符?
            Address?=?(unsigned?char*)MK_FP(DsSelector,OffSetEntry);//合成地址?
            NewValue[0]=0xEA;?
            *((DWORD*)(NewValue+1))?=?(DWORD)MyTextOut;?
            OldValue[
            0]=Address[0];?
            *((DWORD*)(OldValue+1))?=?*((DWORD*)(Address+1));?
            }
            ?
            BOOL?ClearHook()?
            {?
            if(bHookAlready)?
            HookOff();?
            FreeSelector(DsSelector);?
            }
            ?
            BOOL?HookOn()?
            {?
            if(!bHookAlready){?
            for(int?i=0;i<5;i++){?
            Address[i]
            =NewValue[i];?
            }
            ?
            bHookAlready
            =TRUE;?
            }
            ?
            }
            ?
            BOOL?HookOff()?
            {?
            if(bHookAlready){?
            for(int?i=0;i<5;i++){?
            Address[i]
            =OldValue[i];?
            }
            ?
            bHookAlready
            =FALSE;?
            }
            ?
            }
            ?
            //鉤子函數(shù),一定要和API有相同的參數(shù)和聲明?
            BOOL?WINAPI?MyTextOut(HDC?hdc,int?nXStart,int?nYStart,LPCSTR?lpszString,UINT?cbString)?
            {?
            BOOL?ret;?
            HookOff();?
            ret?
            =?TextOut(hdc,nXStart,nYStart,lpszString,cbString);//調(diào)原來的TextOut?
            HookOn();?
            return?ret;?
            }
            ?
            上面的代碼是一個最簡單的掛API鉤子的例子,我要提醒大家的是,這段代碼是我憑記憶寫的,我以前的代碼丟了,我沒有編譯測試過
            因為我沒有VC++1.52.所以代碼可能會有錯。
            建議使用Borland c++,按16位編譯。
            如果用VC++1.52,則要改個選項
            在VC++1.52的Option里,有個內(nèi)存模式的設(shè)置,選大模式,和"DS!=SS DS Load on Function entry.",切記,否則會系統(tǒng)崩潰。
            有什么不明白的可以給我寫信
            yeedong@163.net

            posted on 2006-08-25 00:06 楊粼波 閱讀(587) 評論(0)  編輯 收藏 引用 所屬分類: C++

            久久综合噜噜激激的五月天| 精品久久久久久无码中文野结衣| 人妻丰满AV无码久久不卡| 久久超碰97人人做人人爱| 91精品国产高清91久久久久久 | 亚洲色大成网站www久久九 | 久久本道伊人久久| 午夜精品久久久久9999高清| 成人妇女免费播放久久久| 国产精品热久久无码av| 久久人人爽人人爽人人av东京热 | 亚洲国产成人精品无码久久久久久综合 | 亚洲一级Av无码毛片久久精品| 色88久久久久高潮综合影院 | 亚洲国产精品高清久久久| 国产成人精品综合久久久| 性欧美大战久久久久久久久| 精品久久国产一区二区三区香蕉| 久久久无码一区二区三区| 欧美午夜A∨大片久久| 欧美综合天天夜夜久久| 久久精品www人人爽人人| 国产精品中文久久久久久久 | 青青草国产精品久久| 伊人久久大香线蕉AV色婷婷色| 国产三级精品久久| 亚洲国产精品一区二区久久| 久久亚洲私人国产精品| 久久AV高潮AV无码AV| 午夜视频久久久久一区| 久久久久国产精品嫩草影院| 人人狠狠综合久久亚洲婷婷| 久久精品国产亚洲AV无码麻豆| 久久伊人精品一区二区三区| 亚洲国产成人久久精品99| 久久男人中文字幕资源站| 久久播电影网| 日韩欧美亚洲综合久久影院Ds| 九九久久精品无码专区| 久久国产成人| 久久婷婷五月综合色99啪ak|