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

            大龍的博客

            常用鏈接

            統(tǒng)計

            最新評論

            Windows服務編寫原理及探討(一)

              有那么一類應用程序,是能夠為各種用戶(包括本地用戶和遠程用戶)所用的,擁有用戶授權級進行管理的能力,并且不論用戶是否物理的與正在運行該應用程序的計算機相連都能正常執(zhí)行,這就是所謂的服務了。

            (一)服務的基礎知識

            Question 1. 什么是服務?它的特征是什么?

              在NT/2000中,服務是一類受到操作系統(tǒng)優(yōu)待的程序。一個服務首先是一個Win32可執(zhí)行程序,如果要寫一個功能完備且強大的服務,需要熟悉動態(tài)連接庫(Dlls)、結構異常處理、內存映射文件、虛擬內存、設備I/O、線程及其同步、Unicode以及其他的由WinAPI函數提供的應用接口。當然本文討論的只是建立一個可以安裝、運行、啟動、停止的沒有任何其他功能的服務,所以無需上述知識仍可以繼續(xù)看下去,我會在過程中將理解本文所需要的知識逐一講解。

              第二要知道的是一個服務決不需要用戶界面。大多數的服務將運行在那些被鎖在某些黑暗的,冬暖夏涼的小屋子里的強大的服務器上面,即使有用戶界面一般也沒有人可以看到。如果服務提供任何用戶界面如消息框,那么用戶錯過這些消息的可能性就極高了,所以服務程序通常以控制臺程序的形式被編寫,進入點函數是main()而不是WinMain()

              也許有人有疑問:沒有用戶界面的話,要怎樣設置、管理一個服務?怎樣開始、停止它?服務如何發(fā)出警告或錯誤信息、如何報告關于它的執(zhí)行情況的統(tǒng)計數據?這些問題的答案就是服務能夠被遠程管理,Windows NT/2000提供了大量的管理工具,這些工具允許通過網絡上的其它計算機對某臺機器上面的服務進行管理。比如Windows 2000里面的“控制臺”程序(mmc.exe),用它添加“管理單元”就可以管理本機或其他機器上的服務。

            Question 2. 服務的安全性...

              想要寫一個服務,就必須熟悉Win NT/2000的安全機制,在上述操作系統(tǒng)之中,所有安全都是基于用戶的。換句話說——進程、線程、文件、注冊表鍵、信號、事件等等等等都屬于一個用戶。當一個進程被產生的時候,它都是執(zhí)行在一個用戶的上下文(context),這個用戶帳號可能在本機,也可能在網絡中的其他機器上,或者是在一個特殊的賬號:System Account——即系統(tǒng)帳號的上下文

              如果一個進程正在一個用戶帳號下執(zhí)行,那么這個進程就同時擁有這個用戶所能擁有的一切訪問權限,不論是在本機還是網絡。系統(tǒng)帳號則是一個特殊的賬號,它用來標識系統(tǒng)本身,而且運行在這個帳號下的任何進程都擁有系統(tǒng)上的所有訪問權限,但是系統(tǒng)帳號不能在域上使用,無法訪問網絡資源...

              服務也是Win32可執(zhí)行程序,它也需要執(zhí)行在一個context,通常服務都是在系統(tǒng)賬號下運行,但是也可以根據情況選擇讓它運行在一個用戶賬號下,也就會因此獲得相應的訪問資源的權限。

            Question 3. 服務的三個組成部分

              一個服務由三部分組成,第一部分是Service Control Manager(SCM)。每個Windows NT/2000系統(tǒng)都有一個SCMSCM存在于Service.exe中,在Windows啟動的時候會自動運行,伴隨著操作系統(tǒng)的啟動和關閉而產生和終止。這個進程以系統(tǒng)特權運行,并且提供一個統(tǒng)一的、安全的手段去控制服務。它其實是一個RPC Server,因此我們可以遠程安裝和管理服務,不過這不在本文討論的范圍之內。SCM包含一個儲存著已安裝的服務和驅動程序的信息的數據庫,通過SCM可以統(tǒng)一的、安全的管理這些信息,因此一個服務程序的安裝過程就是將自身的信息寫入這個數據庫。

              第二部分就是服務本身。一個服務擁有能從SCM收到信號和命令所必需的的特殊代碼,并且能夠在處理后將它的狀態(tài)回傳給SCM

              第三部分也就是最后一部分,是一個Service Control Dispatcher(SCP)。它是一個擁有用戶界面,允許用戶開始、停止、暫停、繼續(xù),并且控制一個或多個安裝在計算機上服務的Win32應用程序。SCP的作用是與SCM通訊,Windows 2000管理工具中的“服務”就是一個典型的SCP

              在這三個組成部分中,用戶最可能去寫服務本身,同時也可能不得不寫一個與其伴隨的客戶端程序作為一個SCP去和SCM通訊,本文只討論去設計和實現一個服務,關于如何去實現一個SCP則在以后的其它文章中介紹。

            Question 4. 怎樣開始設計服務

              還記得前面我提到服務程序的入口點函數一般都是main()嗎?一個服務擁有很重要的三個函數,第一個就是入口點函數,其實用WinMain()作為入口點函數也不是不可以,雖然說服務不應該有用戶界面,但是其實存在很少的幾個例外,這就是下面圖中的選項存在的原因。

              由于要和用戶桌面進行信息交互,服務程序有時會以WinMain()作為入口點函數。

              入口函數負責初始化整個進程,由這個進程中的主線程來執(zhí)行。這意味著它應用于這個可執(zhí)行文件中的所有服務。要知道,一個可執(zhí)行文件中能夠包含多個服務以使得執(zhí)行更加有效。主進程通知SCM在可執(zhí)行文件中含有幾個服務,并且給出每一個服務的ServiceMain回調(Call Back)函數的地址。一旦在可執(zhí)行文件內的所有服務都已經停止運行,主線程就在進程終止前對整個進程進行清除。

              第二個很重要的函數就是ServiceMain,我看過一些例子程序里面對自己的服務的進入點函數都固定命名為ServiceMain,其實并沒有規(guī)定過一定要那樣命名,任何的函數只要符合下列的形式都可以作為服務的進入點函數。

            VOID WINAPI ServiceMain(
              DWORD dwArgc, //
            參數個數
              LPTSTR *lpszArgv //
            參數串
            );

              這個函數由操作系統(tǒng)調用,并執(zhí)行能完成服務的代碼。一個專用的線程執(zhí)行每一個服務的ServiceMain函數,注意是服務而不是服務程序,這是因為每個服務也都擁有與自己唯一對應的ServiceMain函數,關于這一點可以用“管理工具”里的“服務”去察看Win2000里面自帶的服務,就會發(fā)現其實很多服務都是由service.exe單獨提供的。當主線程調用Win32函數StartServiceCtrlDispatcher的時候,SCM為這個進程中的每一個服務產生一個線程。這些線程中的每一個都和它的相應的服務的ServiceMain函數一起執(zhí)行,這就是服務總是多線程的原因——一個僅有一個服務的可執(zhí)行文件將有一個主線程,其它的線程執(zhí)行服務本身。

              第三個也就是最后的一個重要函數是CtrlHandler,它必須擁有下面的原型:

            VOID WINAPI CtrlHandler(
            DWORD fdwControl //
            控制命令
            )

              像ServiceMain一樣,CtrlHandler也是一個回調函數,用戶必須為它的服務程序中每一個服務寫一個單獨的CtrlHandler函數,因此如果有一個程序含有兩個服務,那么它至少要擁有5個不同的函數:作為入口點的main()WinMain(),用于第一個服務的ServiceMain函數和CtrlHandler函數,以及用于第二個服務的ServiceMain函數和CtrlHandler函數。

              SCM調用一個服務的CtrlHandler函數去改變這個服務的狀態(tài)。例如,當某個管理員用管理工具里的“服務”嘗試停止你的服務的時候,你的服務的CtrlHandler函數將收到一個SERVICE_CONTROL_STOP通知。CtrlHandler函數負責執(zhí)行停止服務所需的一切代碼。由于是進程的主線程執(zhí)行所有的CtrlHandler函數,因而必須盡量優(yōu)化你的CtrlHandler函數的代碼,使它運行起來足夠快,以便相同進程中的其它服務的CtrlHandler函數能在適當的時間內收到屬于它們的通知。而且基于上述原因,你的CtrlHandler函數必須要能夠將想要傳達的狀態(tài)送到服務線程,這個傳遞過程沒有固定的方法,完全取決于你的服務的用途。

            posted on 2007-12-22 17:55 大龍 閱讀(355) 評論(0)  編輯 收藏 引用

            久久99精品久久久久久hb无码 | 亚洲国产精品高清久久久| 久久久久无码精品| 亚洲AV成人无码久久精品老人| 奇米影视7777久久精品| 久久综合狠狠综合久久激情 | 婷婷久久综合| 久久午夜电影网| 久久精品国产亚洲AV久| 香蕉久久一区二区不卡无毒影院| 久久99国产精品久久99小说| 99精品久久久久中文字幕| 欧美亚洲另类久久综合婷婷| 狠色狠色狠狠色综合久久| 一本色综合久久| 国产精品青草久久久久福利99| 久久妇女高潮几次MBA| 久久精品国产一区二区| 国产一区二区三区久久| 亚洲国产一成人久久精品| 性高朝久久久久久久久久| 93精91精品国产综合久久香蕉| 久久人人爽人人爽人人AV东京热| 国内精品欧美久久精品| 国产精品一久久香蕉产线看| 久久婷婷五月综合97色| 亚洲AV无码久久精品成人| 综合久久给合久久狠狠狠97色 | 99久久亚洲综合精品网站| 色婷婷久久综合中文久久蜜桃av| 午夜精品久久影院蜜桃| 久久精品国产一区二区| 久久久久婷婷| 亚洲精品乱码久久久久久蜜桃| 精品久久人人做人人爽综合| 777久久精品一区二区三区无码| 日本道色综合久久影院| 777久久精品一区二区三区无码| 国产激情久久久久影院小草 | 久久国产精品一国产精品金尊| 久久九九兔免费精品6|