最近的工作需要写一个windows服务,定时执行任务。众所周知windows service程序是没有用户界面的,而有时候是需要知道服务正在做什么的。怎么办呢?可否用winform程序显示服务执行状态呢?如何让winform取到windows service进程的消息呢?这个场景用命名管道不是正合适吗?是的,命名管道正合适。
先来想下实现的思路:
Windows服务程序要声明NamedPipeServerStream对象实例,然后等待winform程序的NamePipeClientStream实例连接到管道;连接到之后windows服务程序将其执行状态通过命名管道输出给winforms程序。
Winform程序可以随时断开管道连接,断开之后windows服务程序要继续等待winform程序再次连接到管道,连接之后可以继续输出其执行状态。另外winform程序可以在局域网的任何一台机器上连接到服务器上的windows服务程序,以方便观察服务的执行状态
思路就是这个样子了,下面我们先看下Windows服务程序中的关键实现:
public class ServiceStatusPublishManager
{
private const int MAX_LENGTH = 200;
private List<string> _msgs;
private ManagerImple _managerImp;
protected ServiceStatusPublishManager()
{
_managerImp = new ManagerImple(this);
_msgs = new List<string>();
}
static public readonly ServiceStatusPublishManager Instance = new ServiceStatusPublishManager();
public void AppendMessage(string msg)
{
lock (this) _msgs.Add(msg);
if (_msgs.Count > MAX_LENGTH)
{
lock (this)
{
_msgs.RemoveAt(0);
if (_managerImp._msgIndex == _msgs.Count) _managerImp._msgIndex -= 2;
}
}
}
class ManagerImple
{
private ServiceStatusPublishManager _trace;
private Thread _thread;
public ManagerImple(ServiceStatusPublishManager trace)
{
_trace = trace;
_thread = new Thread(WriteServiceStatus);
_thread.Start();
}
public int _msgIndex = 0;
private void WriteServiceStatus(object stats)
{
while (true)
{
try
{
using (NamedPipeServerStream serverStream = new NamedPipeServerStream("ServiceStatusPipe", PipeDirection.InOut, 10, PipeTransmissionMode.Message))
{
serverStream.WaitForConnection();
while (serverStream.IsConnected)
{
bool writeSomeMessage = false;
while (_msgIndex < _trace._msgs.Count)
{
string msg = _trace._msgs[_msgIndex];
byte[] bytes = Encoding.UTF8.GetBytes(msg);
serverStream.Write(bytes, 0, bytes.Length);
serverStream.Flush();
_msgIndex++;
writeSomeMessage = true;
}
if (!writeSomeMessage)
{
byte[] bytes = Encoding.UTF8.GetBytes("无消息输出,MsgIndex = " + _msgIndex + ",消息总数 = " + _trace._msgs.Count);
serverStream.Write(bytes, 0, bytes.Length);
serverStream.Flush();
Thread.Sleep(1000 * 10);
}
Thread.Sleep(1000);
}
}
}
catch (Exception ex)
{
}
}
}
}
}
Winform程序中关键实现部分:
void RefreshStatus(object stats)
{
try
{
using (System.IO.Pipes.NamedPipeClientStream stream = new System.IO.Pipes.NamedPipeClientStream(ServerName, "ServiceStatusPipe"))
{
stream.Connect(3000);
stream.ReadMode = System.IO.Pipes.PipeTransmissionMode.Message;
byte[] bytes = new byte[16];
char[] chars = new char[16];
int read;
do
{
string msg = string.Empty;
do
{
read = stream.Read(bytes, 0, bytes.Length);
if (read > 0)
{
int getCharCount = _decoder.GetChars(bytes, 0, read, chars, 0);
msg += new string(chars, 0, getCharCount);
}
} while (!stream.IsMessageComplete);
if (!string.IsNullOrEmpty(msg))
{
AddMsg(msg);
}
_decoder.Reset();
Thread.Sleep(20);
} while (!_closing);
}
}
catch (Exception ex)
{
AddMsg("异常:" + ex.Message);
Thread.Sleep(1000);
RefreshStatus(null);
}
}
前一段时间写过两篇命名管道方面的文章: