Windows Service开发的一个经典有关问题

Windows Service开发的一个经典问题

很多人这样设计Windows Service,希望在某种条件下,比如某个事件发生后,希望Windows Service弹出一个窗体!

其实这非常不正确,不管是C#,VB.net还是C++创建的Windows Service是被OS windows子系统调用的和维护的,

恰恰它没有桌面站,OS在启动Windows子系统的时候,会默认创建两类东西,就是有窗口站的桌面(其实也是一个窗口以及窗口树结构),另外一个就是无窗口站的环境,Windows Service恰恰运行在这个环境中,其含义是没有客户操作区的表面,那么即使你在Windows Service的OnStart方法中new一个Window对象并调用其Show方法,你也看不到有窗体弹出!

 

但是你可以使用一个办法绕过这个限制,那就是使用Windows Service作为服务器端,做一个WCF/Remoting服务,

让一个独立进程(最好系统一起启动的)作为客户端,使用双向回调实现的观察者模式(可以使用委托实现,或者WCF就支持回调的方式,不过需要绑定也支持两个通道,并且回调的行为必须是OneWay的),服务在某个条件下通知被注册的客户端,此时客户端程序就可以有所动作,因为一个exe程序(Console等等)是在有窗口站的环境下运行的,那么它们在接收到服务器通知后就可以弹出一个可以看见的窗体了。

 

 

 

1 楼 ycaicainiao 2011-02-24  
<div class="quote_title">leogao_emcom 写道</div><div class="quote_div"><p>很多人这样设计Windows Service,希望在某种条件下,比如某个事件发生后,希望Windows Service弹出一个窗体!</p>
<p>其实这非常不正确,不管是C#,VB.net还是C++创建的Windows Service是被OS windows子系统调用的和维护的,</p>
<p>恰恰它没有桌面站,OS在启动Windows子系统的时候,会默认创建两类东西,就是有窗口站的桌面(其实也是一个窗口以及窗口树结构),另外一个就是无窗口站的环境,Windows Service恰恰运行在这个环境中,其含义是没有客户操作区的表面,那么即使你在Windows Service的OnStart方法中new一个Window对象并调用其Show方法,你也看不到有窗体弹出!</p>
<p> </p>
<p>但是你可以使用一个办法绕过这个限制,那就是使用Windows Service作为服务器端,做一个WCF/Remoting服务,</p>
<p>让一个独立进程(最好系统一起启动的)作为客户端,使用双向回调实现的观察者模式(可以使用委托实现,或者WCF就支持回调的方式,不过需要绑定也支持两个通道,并且回调的行为必须是OneWay的),服务在某个条件下通知被注册的客户端,此时客户端程序就可以有所动作,因为一个exe程序(Console等等)是在有窗口站的环境下运行的,那么它们在接收到服务器通知后就可以弹出一个可以看见的窗体了。</p>
<p> </p>
<p> </p>
<p> </p></div><br/>记得有一个方法,

http://social.technet.microsoft.com/Forums/zh-CN/w7itproui/thread/9cdc0c96-42e1-4e6d-8f29-78077e87e7b2
2 楼 ycaicainiao 2011-02-24  
<div class="quote_title">leogao_emcom 写道</div><div class="quote_div"><p>很多人这样设计Windows Service,希望在某种条件下,比如某个事件发生后,希望Windows Service弹出一个窗体!</p>
<p>其实这非常不正确,不管是C#,VB.net还是C++创建的Windows Service是被OS windows子系统调用的和维护的,</p>
<p>恰恰它没有桌面站,OS在启动Windows子系统的时候,会默认创建两类东西,就是有窗口站的桌面(其实也是一个窗口以及窗口树结构),另外一个就是无窗口站的环境,Windows Service恰恰运行在这个环境中,其含义是没有客户操作区的表面,那么即使你在Windows Service的OnStart方法中new一个Window对象并调用其Show方法,你也看不到有窗体弹出!</p>
<p> </p>
<p>但是你可以使用一个办法绕过这个限制,那就是使用Windows Service作为服务器端,做一个WCF/Remoting服务,</p>
<p>让一个独立进程(最好系统一起启动的)作为客户端,使用双向回调实现的观察者模式(可以使用委托实现,或者WCF就支持回调的方式,不过需要绑定也支持两个通道,并且回调的行为必须是OneWay的),服务在某个条件下通知被注册的客户端,此时客户端程序就可以有所动作,因为一个exe程序(Console等等)是在有窗口站的环境下运行的,那么它们在接收到服务器通知后就可以弹出一个可以看见的窗体了。</p>
<p> </p>
<p> </p>
<p> </p></div><br/>其实类似杀毒软件都是Windows Service,
明显都是可以直接使用UI的。
3 楼 leogao_emcom 2011-02-24  
引用

记得有一个方法, http://social.technet.microsoft.com/Forums/zh-CN/w7itproui/thread/9cdc0c96-42e1-4e6d-8f29-78077e87e7b2


这只是启动了另外一个进程,不在启动另外一个进程的情况下,能不能?
4 楼 leogao_emcom 2011-02-24  
杀毒软件也是间接的,用类似通信的办法绕过windows service这个限制的,而不是直接。
5 楼 leogao_emcom 2011-02-24  
public partial class Service1 : ServiceBase
    {
        public Service1()
        {
            InitializeComponent();
        }

        protected override void OnStart(string[] args)
        {
           
            Form ooo = new Form();
            ooo.Show();
        }

        protected override void OnStop()
        {

        }
    }
使用installutil安装一下,并且attach到debugger上,debug一下,窗体果然是不能跳出来显示的,即使是release模式build后,安装,start,也是不行的。我说的就是这样的情况
6 楼 wumingshi 2011-02-24  
这样的描述并不正确。windowsservice是可以与桌面交互的。但是必须在service安装时添加与桌面交互选项。或者手工在service的属性里修改。然后才有可能进行ui操作。当然楼主分离ui的思路是好的,因为没有ui的服务会降低内存消耗。
7 楼 leogao_emcom 2011-02-25  
即使在安装时添加与桌面交互选项,也没有效果,已经试过了。
8 楼 leogao_emcom 2011-02-25  
引用

windowsservice是可以与桌面交互的。但是必须在service安装时添加与桌面交互选项


这句话应该修正为,在开发模式下,在为Service添加的ServiceProcessInstaller属性中的Account设置为LocalSystem,然后编译,使用installutil进行安装到windows Service编录(管理器),然后在对应的服务属性中【登录】页中,设置本地系统账户的【允许与桌面交互】选项被选中,最后启动服务,此时桌面就会显示出来。
9 楼 leogao_emcom 2011-02-25  
哇哇,窗体是出来了,但是在UI上无法进行操作?我没有找到问题所在。需要再继续探讨一下
10 楼 leogao_emcom 2011-02-25  
OK!问题解决,不使用show方法,而是ShowDialog方法!
public partial class Service1 : ServiceBase
    {
        public Service1()
        {
            InitializeComponent();
        }

        protected override void OnStart(string[] args)
        {

            Form ooo = new Form();
            ooo.ShowDialog();
        }

        protected override void OnStop()
        {

        }
    }
再加上:在开发模式下,在为Service添加的ServiceProcessInstaller属性中的Account设置为LocalSystem,然后编译,使用installutil进行安装到windows Service编录(管理器),然后在对应的服务属性中【登录】页中,设置本地系统账户的【允许与桌面交互】选项被选中,最后启动服务,此时桌面就会显示出来。并且UI可以正常操作了。
11 楼 leogao_emcom 2011-02-25  
正解!我看此课题可以结案了。good!感谢大家