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

            悲情土仔一生

              C++博客 :: 首頁(yè) :: 新隨筆 :: 聯(lián)系 :: 聚合  :: 管理 ::
              14 隨筆 :: 0 文章 :: 74 評(píng)論 :: 0 Trackbacks

            利用未公開(kāi)API獲取終端會(huì)話閑置時(shí)間(Idle Time)和登入時(shí)間(Logon Time)
            作者:Tuuzed(土仔)   發(fā)表于:2008年3月3日23:12:38 
            版權(quán)聲明:可以任意轉(zhuǎn)載,轉(zhuǎn)載時(shí)請(qǐng)務(wù)必以超鏈接形式標(biāo)明文章原始出處和作者信息及本聲明
            http://www.shnenglu.com/tuuzed/archive/2008/03/03/43631.html


                
                可能很多人都知道NT系統(tǒng)的query user命令,命令返回“使用者名稱(chēng) 工作階段名稱(chēng) 識(shí)別碼 狀態(tài) 閑置時(shí)間 登入時(shí)間”。如圖:

            TS0

                微軟給出了獲取終端會(huì)話的重要API(見(jiàn)Terminal Services API Functions),與獲取當(dāng)前終端會(huì)話功能有關(guān)的API有:WTSEnumerateSessionsWTSQuerySessionInformation

                    WTSEnumerateSessions:顧名思義就是列出所有的Session,返回一個(gè)WTS_SESSION_INFO結(jié)構(gòu),結(jié)構(gòu)存儲(chǔ)了SessionId,WinStationName,State(包括Active、Disconnected等狀態(tài))。

                    WTSQuerySessionInformation:這個(gè)和上面的API有些不同,它只能通過(guò)SessionId來(lái)查詢(xún)Session的詳細(xì)信息,可獲取例如用于連接終端客戶(hù)端工具的ClientName、ClientDirectory等,比WTSEnumerateSessions功能豐富。

                按照MSDN上說(shuō)的,WTSQuerySessionInformation還可以獲取IdleTime、LogonTime、IncomingBytes、OutgoingBytes等信息,可惜,標(biāo)明是“This value is not used.”,要使用的話必須在Windows Server 2008和Windows Vista SP1下使用,局限性太大了。只好自己上Goolge上搜索一下了,在國(guó)外的論壇中,大部分人對(duì)于獲取Idle Time都是說(shuō)在WIN2008或VISTA才支持。那么WIN2000、2003里的query命令是怎么獲得登入時(shí)間的?這里面肯定有什么沒(méi)有公開(kāi)的API在里面!果然,我找到了Guy Teverovsky的BLOG,它給出的答案(《Querying TS session idle time with C#》譯文:《[翻譯]利用C#獲取終端服務(wù)(Terminal Services)會(huì)話的閑置時(shí)間》)和我預(yù)想的差不錯(cuò)——所要的信息在在Winsta.dll內(nèi)的一個(gè)未公開(kāi)API函數(shù)WinStationQueryInformationW返回的結(jié)構(gòu)WINSTATIONQUERYINFORMATIONW里面。

                要想使用WinStationQueryInformationW必須知道其中兩個(gè)重要的參數(shù)WinStationInformation(枚舉類(lèi)型)值和WINSTATIONINFORMATIONW結(jié)構(gòu)內(nèi)容。在VS2005對(duì)上述兩個(gè)值有定義(winternl.h):

             

            typedef enum _WINSTATIONINFOCLASS {
                WinStationInformation 
            = 8
            }
             WINSTATIONINFOCLASS; 

             

            typedef struct _WINSTATIONINFORMATIONW {
                BYTE Reserved2[
            70];
                ULONG LogonId;
                BYTE Reserved3[
            1140];
            }
             WINSTATIONINFORMATIONW, * PWINSTATIONINFORMATIONW; 

             

                第一個(gè)值很清楚了,是8。而后一個(gè)結(jié)構(gòu),保留位達(dá)1140位,這里有太多未知的信息了。還好那位牛人給出了C#的定義,我把它轉(zhuǎn)成C++的結(jié)構(gòu)定義:

            typedef struct _WINSTATIONQUERYINFORMATION
            {
                
            char Reserved1[72];
                unsigned 
            int SessionId;
                
            char Reserved2[4];
                FILETIME ConnectTime;
                FILETIME DisconnectTime;
                FILETIME LastInputTime;
                FILETIME LogonTime;
                
            char Reserved3[1096];
                FILETIME CurrentTime;
            }
             WINSTATIONQUERYINFORMATION, *PWINSTATIONQUERYINFORMATION; 

             

                定義完這個(gè)結(jié)構(gòu),工作已經(jīng)有眉目了。下面就是載入Winsta.dll內(nèi)那個(gè)未公開(kāi)的API函數(shù),順便包裝了一下:

            BOOL WINAPI WinStationQueryInformation(HANDLE hServer, DWORD SessionId, DWORD InfoClass, LPVOID Buffer, DWORD BufferLength, LPDWORD Count)
            {
                typedef BOOL (WINAPI 
            *PROCPTR)(HANDLE, DWORD, DWORD, LPVOID, DWORD, LPDWORD);
                
            static HMODULE hModule = NULL;
                
            static PROCPTR proc = NULL;
                hModule 
            = LoadLibrary("winsta.dll");
                
            if (hModule == NULL)
                
            {
                    
            return FALSE;
                }
             

                
            if (proc == NULL)
                
            {
                    proc 
            = (PROCPTR) GetProcAddress(hModule, "WinStationQueryInformationW");
                }
             

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

                
            return proc(hServer, SessionId, InfoClass, Buffer, BufferLength, Count);
            }
             

             

                這樣,只要直接調(diào)用自己的WinStationQueryInformation來(lái)間接調(diào)用DLL里面的WinStationQueryInformationW就可以了。登入時(shí)間Logon Time是可以直接獲取的,而閑置時(shí)間的獲取就要參考當(dāng)前會(huì)話的狀態(tài)了:如果會(huì)話是斷開(kāi)(Disconnected)狀態(tài),閑置時(shí)間=當(dāng)前時(shí)間-斷開(kāi)時(shí)間(Idle Time = CurrentTime - DisconnectTime);如果會(huì)話是活動(dòng)的(alive)狀態(tài),閑置時(shí)間=當(dāng)前時(shí)間-最后輸入時(shí)間(Idle Time = CurrentTime - LastInputTime)。

                已經(jīng)做出來(lái)了一個(gè)DEMO,看截圖:

            TS1

                這是我做的一個(gè)ROOKIT工具的截圖,其中就包括有對(duì)終端會(huì)話的管理等。在這次編程后,給我感觸最深的就是國(guó)外大蝦解決問(wèn)題的方法與國(guó)內(nèi)的我們有很大的不同,他們善于從多角度來(lái)解決問(wèn)題,獨(dú)自解決問(wèn)題后再總結(jié),是利用網(wǎng)絡(luò)資源而不是依賴(lài)網(wǎng)絡(luò)資源。

            posted on 2008-03-03 22:51 土仔 閱讀(8431) 評(píng)論(7)  編輯 收藏 引用 所屬分類(lèi): 土仔編程

            評(píng)論

            # re: 利用未公開(kāi)API獲取終端會(huì)話閑置時(shí)間(Idle Time)和登入時(shí)間(Logon Time) 2008-03-04 09:51 jaychung
            不錯(cuò)  回復(fù)  更多評(píng)論
              

            # re: 利用未公開(kāi)API獲取終端會(huì)話閑置時(shí)間(Idle Time)和登入時(shí)間(Logon Time) 2008-03-04 16:49 菌子
            永遠(yuǎn)不要使用未公開(kāi)API.   回復(fù)  更多評(píng)論
              

            # re: 利用未公開(kāi)API獲取終端會(huì)話閑置時(shí)間(Idle Time)和登入時(shí)間(Logon Time)[未登錄](méi) 2008-04-20 17:33 Toto
            永遠(yuǎn)不要使用未公開(kāi)API--永遠(yuǎn)被微軟牽著鼻子走  回復(fù)  更多評(píng)論
              

            # re: 利用未公開(kāi)API獲取終端會(huì)話閑置時(shí)間(Idle Time)和登入時(shí)間(Logon Time) 2008-06-25 16:13 笨笨牛
            為什么不提供源碼呢  回復(fù)  更多評(píng)論
              

            # re: 利用未公開(kāi)API獲取終端會(huì)話閑置時(shí)間(Idle Time)和登入時(shí)間(Logon Time) 2009-06-01 20:26 百變貍貓
            使用未公開(kāi)的API有很大風(fēng)險(xiǎn).
            大的公司做軟件一般不會(huì)采用
            因?yàn)闀?huì)有不同版本兼容性問(wèn)題等各種問(wèn)題  回復(fù)  更多評(píng)論
              

            # re: 利用未公開(kāi)API獲取終端會(huì)話閑置時(shí)間(Idle Time)和登入時(shí)間(Logon Time) 2013-04-20 23:32 re
            沒(méi)怎么看明白,BOOL WINAPI WinStationQueryInformation(HANDLE hServer, DWORD SessionId, DWORD InfoClass, LPVOID Buffer, DWORD BufferLength, LPDWORD Count)
            這個(gè)在調(diào)用的時(shí)候參數(shù)都是什么,希望高人給解釋下,郵箱:654054668@qq.com 在此多謝了  回復(fù)  更多評(píng)論
              

            # re: 利用未公開(kāi)API獲取終端會(huì)話閑置時(shí)間(Idle Time)和登入時(shí)間(Logon Time) 2015-11-26 16:19 東風(fēng)破
            WinStationQueryInformation傳入WTSSessionInfo完全可以獲取到LoginTime和Idle Time,7年了,可能當(dāng)時(shí)還沒(méi)這個(gè)參數(shù)吧。  回復(fù)  更多評(píng)論
              

            久久亚洲天堂| 久久久久99这里有精品10 | 国产精品久久波多野结衣| 久久精品国产亚洲av瑜伽| 91精品国产综合久久四虎久久无码一级| 国产成人精品综合久久久| 国产99久久久国产精品小说| 亚洲国产精品成人久久蜜臀| 日韩va亚洲va欧美va久久| 欧美激情精品久久久久久| 久久精品国产99久久久香蕉| 久久亚洲中文字幕精品一区四| 久久久久无码国产精品不卡| 性做久久久久久久久老女人| 成人综合久久精品色婷婷| 久久久久av无码免费网| 久久久久人妻一区精品色| 国产精品无码久久综合| 久久免费美女视频| 日日狠狠久久偷偷色综合免费| 国产精品久久久久久久久软件| 人妻少妇久久中文字幕一区二区| 国产精品美女久久久久网| 久久久久久无码国产精品中文字幕 | 香蕉久久夜色精品升级完成| 97久久精品人妻人人搡人人玩| 国内精品久久久久久野外| 久久99精品久久久久久不卡 | 久久只有这里有精品4| 久久久久久久亚洲Av无码| 99久久综合狠狠综合久久| 亚洲?V乱码久久精品蜜桃| 日韩av无码久久精品免费| 久久久久人妻一区精品| 国产精品一区二区久久精品| 日本精品久久久久影院日本| 97热久久免费频精品99| 无码8090精品久久一区| 国产精品久久网| 亚洲va久久久噜噜噜久久男同 | 深夜久久AAAAA级毛片免费看|