从Window系统托盘控制Windows服务

net_lover(原作)
作者: 孟宪会 出自: 【孟宪会之精彩世界】 发布日期: 2003-8-5 17:38:11

VB.NET有许多内建的类,可以让我们轻松创建Windows服务程序,但如何轻松控制这些服务呢?一般是到管理工具里面进行控制。本文将描述如何创建一个运行在系统托盘里的程序来轻松控制一个服务程序。至于如何创建服务程序,可以参考.NET SDK或其它创建服务程序的文章,本文的例子利用IIS的W3SVC服务来做例子,来控制IIS的停止与启动。

要开发这样的程序,我们先打开Microsoft Visual Studio.NET,新建一个名为ServiceController的解决方案,然后新建名为WinForm的Visual Basic类型的Windows应用程序,然后把VS.NET自动创建的Form1.vb删除掉,因为我们创建的应用程序没有用户界面,我们在Sub Main运行程序。

先添加引用-.NET-System.ServiceProcess.dll,新建名为modMain的模块,代码如下:

Imports System.Text
Imports System.Diagnostics
Imports System.ServiceProcess

Public Module modMain

  Private WithEvents mobNotifyIcon As NotifyIcon
  Private WithEvents mobContextMenu As ContextMenu
  Private WithEvents mobTimer As Timers.Timer
  Private mobServiceController As ServiceController

End Module

上面的代码首先引用了三个名称空间,然后分别定义了四个变量:mobNotifyIcon将会在系统托盘里显示;ContextMenu显示菜单信息;mobTimer为定时器,原来检查服务的状态,以随时变更菜单和图标的状态;mobServiceController表示Windows服务应用程序并允许连接到正在运行或者已停止的服务、对其进行操作或获取有关它的信息。

由于服务程序是没有用户界面的,因此我们设置三种图标标识服务的状态,这里做了三个简单的图标来标识服务的状态:Running.ico,Paused.ico,Stopped.ico,分别如下:

pst21_1

下面我们就建立定时器SetUpTimer过程,通常,IIS停止或启动的间隔为5秒,我们就用5秒来做定时器的间隔,代码如下:

Private Sub SetUpTimer()
    Try
        mobTimer = New Timers.Timer()
        With mobTimer
            .AutoReset = True
            .Interval = 5000
            .Start()
        End With
    Catch obEx As Exception
        Throw obEx
    End Try
End Sub

下面,创建上下文菜单的过程,并为每个菜单项添加事件处理程序:

Private Sub CreateMenu()
    Try
        mobContextMenu.MenuItems.Add(New MenuItem("停止",New EventHandler(AddressOf StopService)))
        mobContextMenu.MenuItems.Add(New MenuItem("暂停",New EventHandler(AddressOf PauseService)))
        mobContextMenu.MenuItems.Add(New MenuItem("继续",New EventHandler(AddressOf ContinueService)))
        mobContextMenu.MenuItems.Add(New MenuItem("开始",New EventHandler(AddressOf StartService)))
        mobContextMenu.MenuItems.Add("-")
        mobContextMenu.MenuItems.Add(New MenuItem("关于",New EventHandler(AddressOf AboutBox)))
        mobContextMenu.MenuItems.Add(New MenuItem("退出",New EventHandler(AddressOf ExitController)))
    Catch obEx As Exception
        Throw obEx
    End Try
End Sub

当我们改变了服务的运行状态时,我们应当向用户反映这一变化,这里用托盘的图标不同来进行标识。当服务程序启动时,就要先建立服务的状态,首先GetServiceStatus过程调用ServiceController的Refresh方法,它将会刷新的ServiceController所有属性。要准确得到服务程序的状态,这一过程是至关重要的,下面的Select Case语句根据不同的服务程序的状态,建立不同的菜单项和托盘图标。

Private Sub GetServiceStatus()
    Try
    '//读取状态之前先进行刷新
    mobServiceController.Refresh()
    '//变更菜单项和图标
    Select Case mobServiceController.Status()
          Case ServiceProcess.ServiceControllerStatus.Paused
          mobNotifyIcon.Icon = New Icon("Paused.ico")
          mobContextMenu.MenuItems(0).Enabled = False
          mobContextMenu.MenuItems(1).Enabled = False
          mobContextMenu.MenuItems(2).Enabled = True
          mobContextMenu.MenuItems(3).Enabled = False
          Case ServiceProcess.ServiceControllerStatus.Running
          mobNotifyIcon.Icon = New Icon("Running.ico")
          mobContextMenu.MenuItems(0).Enabled = True
          mobContextMenu.MenuItems(1).Enabled = True
          mobContextMenu.MenuItems(2).Enabled = False
          mobContextMenu.MenuItems(3).Enabled = False
          Case ServiceProcess.ServiceControllerStatus.Stopped
          mobNotifyIcon.Icon = New Icon("Stopped.ico")
          mobContextMenu.MenuItems(0).Enabled = False
          mobContextMenu.MenuItems(1).Enabled = False
          mobContextMenu.MenuItems(2).Enabled = False
          mobContextMenu.MenuItems(3).Enabled = True
          Case _
          ServiceProcess.ServiceControllerStatus.ContinuePending, _
              ServiceProcess.ServiceControllerStatus.PausePending, _
              ServiceProcess.ServiceControllerStatus.StartPending, _
              ServiceProcess.ServiceControllerStatus.StopPending
          mobNotifyIcon.Icon = New Icon("Paused.ico")
          mobContextMenu.MenuItems(0).Enabled = False
          mobContextMenu.MenuItems(1).Enabled = False
          mobContextMenu.MenuItems(2).Enabled = False
          mobContextMenu.MenuItems(3).Enabled = False
    End Select
    '//检查“暂停”和“继续”使用可用
    If mobServiceController.CanPauseAndContinue = False Then
        mobContextMenu.MenuItems(1).Enabled = False
        mobContextMenu.MenuItems(2).Enabled = False
    End If

    Catch obEx As Exception
        Throw obEx
    End Try
End Sub

下面建立菜单项的事件处理程序:

'//停止服务的过程
Private Sub StopService(ByVal sender As Object, ByVal e As EventArgs)
    Try
        If mobServiceController.Status = ServiceProcess.ServiceControllerStatus.Running Then
            If mobServiceController.CanStop = True Then
                mobServiceController.Stop()
            End If
        End If
    Catch obEx As Exception
        Throw obEx
    End Try
End Sub

'//暂停服务的过程
Private Sub PauseService(ByVal sender As Object, ByVal e As EventArgs)
    Try
        If Not mobServiceController.Status = ServiceProcess.ServiceControllerStatus.Paused = True Then
            If mobServiceController.CanPauseAndContinue = True Then
                mobServiceController.Pause()
            End If
        End If
    Catch obEx As Exception
        Throw obEx
    End Try
End Sub

'//继续服务程序的过程
Private Sub ContinueService(ByVal sender As Object, ByVal e As EventArgs)
    Try
        If mobServiceController.Status = ServiceProcess.ServiceControllerStatus.Paused = True Then
            If mobServiceController.CanPauseAndContinue = True Then
                mobServiceController.Continue()
            End If
        End If
    Catch obEx As Exception
        Throw obEx
    End Try
End Sub

'//开始服务程序的过程
Private Sub StartService(ByVal sender As Object, ByVal e As EventArgs)
        Try
        If mobServiceController.Status = ServiceProcess.ServiceControllerStatus.Stopped Then
            mobServiceController.Start()
        End If
    Catch obEx As Exception
        Throw obEx
    End Try
End Sub

'//“关于”菜单项的过程
Private Sub AboutBox(ByVal sender As Object, ByVal e As EventArgs)
    Try
        Dim obStringBuilder As New StringBuilder()
        With obStringBuilder
            .Append("Service Controller 使用例子")
            .Append(vbCrLf)
            .Append("CLR 版本:")
            .Append(Environment.Version.ToString)
            MsgBox(.ToString, MsgBoxStyle.Information)
        End With
        obStringBuilder = Nothing
    Catch obEx As Exception
        Throw obEx
    End Try
End Sub

'//退出服务程序的过程
Private Sub ExitController(ByVal sender As Object, ByVal e As EventArgs)
    Try
        mobTimer.Stop()
        mobTimer.Dispose()
        mobNotifyIcon.Visible = False
        mobNotifyIcon.Dispose()
        mobServiceController.Dispose()
        Application.Exit()
    Catch obEx As Exception
        Throw obEx
    End Try
End Sub

'//定时器停止
Public Sub mobTimer_Elapsed(ByVal sender As Object, ByVal e As System.Timers.ElapsedEventArgs) _
    Handles mobTimer.Elapsed
    Try
        GetServiceStatus()
    Catch obEx As Exception
        Throw obEx
    End Try
End Sub

'//系统托盘图标单击事件
Public Sub mobNotifyIcon_Click(ByVal sender As Object, ByVal e As System.EventArgs) _
    Handles mobNotifyIcon.Click
    System.Diagnostics.Process.Start("IExplore.exe", "http://xml.sz.luohuedu.net/")
End Sub

下面就是主程序:

Public Sub Main()
    Try
        '//建立与服务程序的连接
        mobServiceController = New System.ServiceProcess.ServiceController("IISAdmin")
        '//隐藏图标,知道菜单项和图标准备好以后。
        mobNotifyIcon = New NotifyIcon()
        mobNotifyIcon.Visible = False
        mobContextMenu = New ContextMenu()
        CreateMenu()
        mobNotifyIcon.ContextMenu = mobContextMenu
        mobNotifyIcon.Text = "【孟宪会之精彩世界】" + _
            Microsoft.VisualBasic.ChrW(10) + "http://xml.sz.luohuedu.net/"
        SetUpTimer()
        mobNotifyIcon.Visible = True
        Application.Run()
    Catch obEx As Exception
        MsgBox(obEx.Message.ToString, MsgBoxStyle.Critical)
        End
    End Try
End Sub

运行结果如下:

pts21_2


liuguangwei (2003-10-7 11:17:47)

很不错的程序!值得学习.

ceocio (2003-8-24 10:57:28)

孟子出马,果然不同凡响!:)

Contributors: FHL