此文档假设你已会C和C#的服务编程了。
以前我在写服务程序的时候,SCM有一个很酷的功能,能接收WinLogon的Notify消息。具体操作是,设置服务状态的时候,把SERVICE_STATUS结构体变量的dwControlsAccepted参数设成包含SERVICE_ACCEPT_SESSIONCHANGE的异或(代码一),然后就能在HandleEx函数里处理了(代码2)。你可以方便地获得系统的Logon、LogOff、Lock、Unlock的消息,并在此时机做出自己的处理。例如开始菜单里面"启动"项里的程序,如果你要模拟实现会有很多种方法,服务就是其中的一种。再比如你要在系统登出的时候做一些处理,服务几乎是最佳的选择。
代码一:
MyServiceStatus.dwControlsAccepted = SERVICE_ACCEPT_STOP | SERVICE_ACCEPT_SHUTDOWN|
SERVICE_ACCEPT_PAUSE_CONTINUE | SERVICE_ACCEPT_SESSIONCHANGE;
代码2:
VOID HandlerEx(DWORD controlCode,
DWORD dwEventType,
LPVOID lpEventData,
LPVOID lpContext)
{
switch(controlCode)
{
case SERVICE_CONTROL_SESSIONCHANGE:
switch(dwEventType)
{
case WTS_SESSION_LOGOFF:
// Logoff
break;
case WTS_SESSION_LOGON:
//LogOn Message
break;
case WTS_SESSION_LOCK:
//Lock Message
break;
case WTS_SESSION_UNLOCK:
//Unlock Message
break;
default:
break;
}
break;
}
后来我碰巧接手了一个C#的服务,我第一次看到用C#写的服务,大吃一惊,像外星人见到了地球人一样!怎么有这么怪的东西啊!没有HandleEx函数,取而代之的是一个个OnStart、OnStop、OnShutDown函数。我试着寻找有没有类似OnSessionChange的函数,在.net 2.0里面被我找到了,而奇怪的是,.net 1.0里面竟然没有,并且即便2.0里面有此函数,它所支持的系统竟然不包括Win2000!这足以说明.net的体质不健康,和"比较"畸形!对于第一个1.1不支持的问题,我的一位同事曾尝试着用重载OnCustomCommand的方法来模拟实现,他用反编译工具看源代码,发现1.1基类里面ServiceBase::OnCustomCommand调用了另一个"保护"方法,而这个方法对OnSessionChage消息不做处理,忽视,从而使得C#的服务从根本上不能处理WinLogOn消息--基类的保护方法不能重载(注意WinLogOn消息是一定能够接受到的,因为消息的发送仅仅由SCM来决定,它是做群发的)。
好,现在来看看2.0里面应该怎么做吧!
好像在C服务里面设置dwControlsAccepted一样,在InitializeComponent()方法里面设置类变量的值:this.CanHandleSessionChangeEvent = true;这样服务接收SessionChage的功能就有效了。下面来看接收WinLogOn消息的实现代码,OnSessionChage函数:
代码三:
protectedoverridevoid OnSessionChange(SessionChangeDescription Description)
{
try
{
switch(Description.Reason)
case SessionChangeReason.SessionLogoff:
//
break;
default:
break;
}
catch (Exception ex)
{
Log.Error(func + ex.Message + "\r\n" + ex.StackTrace);
}
}
同样,你可以在接受到消息的时候做许多你想要做的事情。注意,此功能你不能用于Win2K Pro。