文档章节

C#WINDOWS服务交互的实现

短短的歼击机
 短短的歼击机
发布于 2012/11/29 11:21
字数 955
阅读 126
收藏 0

C#WINDOWS服务交互的实现的前言:这几天想做个文件监控服务,看了一下网上的关于WINDOWS服务的文章,数量都不少,都只讲了如何做一个最基本的服务,却没有讲述如何与用户进行交互。查看了MSDN,看一下关于服务的描述:

WINDOWS服务交互应用程序在不同于登录用户的交互区域的窗口区域中运行。窗口区域是包含剪贴板、一组全局原子和一组桌面对象的安全对象。由于 WINDOWS服务交互的区域不是交互区域,因此 Windows 服务应用程序中引发的对话框将是不可见的,并且可能导致程序停止响应。同样,错误信息应记录在 Windows 事件日志中,而不是在用户界面中引发。

.NET Framework 支持的 WINDOWS服务交互类不支持与交互区域(即登录用户)进行交互。同时,.NET Framework 不包含表示区域和桌面的类。如果 WINDOWS服务交互务必须与其他区域进行交互,则需要访问非托管的 Windows API。

也就是说我们要实现可交互的服务(比如我们想给服务在运行时做一些参数设置等),那我们一定要using System.Runtime.InteropServices

那么来看一下如果才能实现一个可交互的服务呢。步骤与实现基本的服务一样(各位可自行参考MSDN或网上google一下).

在实现OnStart时要注意,这里可不能弹出一个FORM什么的。这样做是没有任何反应的。我们可以在这个方法里运行一个线程。该线程需要访问窗口区域对象或桌面对象,当然 framework里是没有提供这些的,要访问非托管代码的。

来看一下代码,再运行试一下。

using System;  
    using System.Collections;  
    using System.ComponentModel;  
    using System.Data;  
    using System.Diagnostics;  
    using System.ServiceProcess;  
    using System.Threading;  
    using System.Runtime.InteropServices;  
    namespace FileWatchService  
    {  
    public class Service1 : System.ServiceProcess.ServiceBase  
    {  
    ///  
    /// 必需的设计器变量。 
    ///  
    private System.ComponentModel.Container components = null;  
    Thread threadForm=null;  
    public Service1()  
    {  
    // 该调用是 Windows.Forms 组件设计器所必需的。 
    InitializeComponent();  
     
    // TODO: 在 InitComponent 调用后添加任何初始化 
    } 
     
    #region 组件设计器生成的代码 
    ///  
    /// 设计器支持所需的方法 - 不要使用代码编辑器  
    /// 修改此方法的内容。 
    ///  
    private void InitializeComponent()  
    {  
    //  WINDOWS服务交互
    // Service1 
    //  
    this.ServiceName = "JadeWatchService";  
     
    } 
    #endregion 
    [STAThread]  
    static void Main()   
    {  
    System.ServiceProcess.ServiceBase.Run(new Service1());  
     
    }  
    ///  WINDOWS服务交互
    /// 清理所有正在使用的资源。 
    ///  
    protected override void Dispose( bool disposing )   
    {  
    if( disposing )  
    {  
    if (components != null)   
    {  
    components.Dispose();  
    }  
    }  
    base.Dispose( disposing );  
    }  
     
    ///  
    /// 设置具体的操作,以便服务可以执行它的工作。 
    ///  
    protected override void OnStart(string[] args)  
    {  
    threadForm=new Thread(new ThreadStart(FormShow));  
    threadForm.Start();  
    }  
     
    ///  WINDOWS服务交互
    /// 停止此服务。 
    ///  
    protected override void OnStop()  
    {  
    if(threadForm!=null)  
    {  
    if(threadForm.IsAlive)  
    {  
    threadForm.Abort();  
    threadForm=null;  
    }  
    }  
    }  
     
    void FormShow()  
    {  
     
    GetDesktopWindow();   
    IntPtr hwinstaSave = GetProcessWindowStation();   
    IntPtr dwThreadId = GetCurrentThreadId();   
    IntPtr hdeskSave = GetThreadDesktop(dwThreadId);   
    IntPtr hwinstaUser = OpenWindowStation(
    "WinSta0", false,33554432);   
    if (hwinstaUser == IntPtr.Zero)   
    {   
    RpcRevertToSelf();   
    return ;  
    }   
    SetProcessWindowStation(hwinstaUser);   
    IntPtr hdeskUser = OpenDesktop(
    "Default", 0, false, 33554432);   
    RpcRevertToSelf();   
    if (hdeskUser == IntPtr.Zero)   
    {   
    SetProcessWindowStation(hwinstaSave);   
    CloseWindowStation(hwinstaUser);   
    return ;   
    }   
    SetThreadDesktop(hdeskUser);   
     
    IntPtr dwGuiThreadId = dwThreadId;   
     
    Form1 f=new Form1(); 
    //此FORM1可以带notifyIcon,可以显示在托盘里,用户可点击托盘图标进行设置 
    System.Windows.Forms.Application.Run(f);  
     //WINDOWS服务交互
     
    dwGuiThreadId = IntPtr.Zero;   
    SetThreadDesktop(hdeskSave);   
    SetProcessWindowStation(hwinstaSave);   
    CloseDesktop(hdeskUser);   
    CloseWindowStation(hwinstaUser);   
    }  
     
    [DllImport("user32.dll")]  
    static extern int GetDesktopWindow();  
     
    [DllImport("user32.dll")]  
    static extern IntPtr GetProcessWindowStation();  
     
    [DllImport("kernel32.dll")]  
    static extern IntPtr GetCurrentThreadId();  
     
    [DllImport("user32.dll")]  
    static extern IntPtr GetThreadDesktop(IntPtr dwThread);  
     
    [DllImport("user32.dll")]  
    static extern IntPtr OpenWindowStation(string a,bool b,int c);  
     
    [DllImport("user32.dll")]  
    static extern IntPtr OpenDesktop(string lpszDesktop, uint dwFlags,  
    bool fInherit, uint dwDesiredAccess);  
     
    [DllImport("user32.dll")]  
    static extern IntPtr CloseDesktop(IntPtr p);  
     
    [DllImport("rpcrt4.dll", SetLastError=true)]  
    static extern IntPtr RpcImpersonateClient(int i);  
     
     
    [DllImport("rpcrt4.dll", SetLastError=true)]  
    static extern IntPtr RpcRevertToSelf();  
     
    [DllImport("user32.dll")]  
    static extern IntPtr SetThreadDesktop(IntPtr a);  
     
    [DllImport("user32.dll")]  
    static extern IntPtr SetProcessWindowStation(IntPtr a);  
    [DllImport("user32.dll")]  
    static extern IntPtr CloseWindowStation(IntPtr a);  
    }  
    }

C#WINDOWS服务交互的实现的相关内容就向你介绍到这里,希望对你学习和了解C#WINDOWS服务交互的实现有所帮助。

© 著作权归作者所有

短短的歼击机

短短的歼击机

粉丝 82
博文 268
码字总数 269797
作品 0
武汉
高级程序员
私信 提问
C#Windows服务程序开发实例浅析

C#Windows服务程序开发实例:编写一个C#Windows服务程序,定时从数据库中拿出记录发送邮件。 测试环境:Visual Studio 2005 SP1、Windows Server 2003 SP2 C#Windows服务程序开发实例一、新建...

postdep
2012/11/29
77
0
C#windows服务中的Timer控件的使用

C# windows服务程序中的Timer控件的使用问题是如何解决的呢? 今天和同事一起研究了下C# windows服务程序中的Timer控件的使用的写法. 我们在建立一个C# windows服务程序后,要用到计时器的话一...

postdep
2012/11/29
123
0
为.Net程序集添加资源

作者:宋华 本文选自:赛迪网 2002年12月06日 在.Net结构中,程序集Assembly(也称着程序集或托管Dll)是自我描述的安装单元,它可以只包括一个PE(可移植可执行)格式的Dll或exe文件,也可以由...

晨曦之光
2012/03/09
168
0
2016/09/13 罗总

为什么做iSR平台的思考 每年做业务规划都是同样的内容,没有考虑到发展的趋势,和互联网的影响。服务模式的变化,带来经营模式的变化。 iSP平台,如何利用互联网的工具,服务模式的创新,做好...

Frank_Zhang
2016/09/13
1
0
使用 SoaML 服务架构

架构概念是面向服务架构(SOA)封装参与者之间交互的基础。本文探讨了如何使用 OMG SoaML 标准对服务架构进行建模。 简介 OMG SoaML 规范引入了服务架构的概念,是用于对一组参与者如何通过所...

IBMdW
2012/10/16
364
0

没有更多内容

加载失败,请刷新页面

加载更多

Linux 之docker部署,走过的那些坑

初次使用docker, 多有不足,这里把坑一点点的记下来 概念篇 docker 是什么? 太多文字不同,书读的少, 不懂大大道理; 在我的理解, 它就是个沙箱环境; 在linux中 ,独立进程, 有着自己的小世界 使...

莫库什勒
14分钟前
2
0
vue class绑定 组件

本文转载于:专业的前端网站➬vue class绑定 组件 当在一个自定义组件上使用 class 属性时,这些类将被添加到该组件的根元素上面。这个元素上已经存在的类不会被覆盖。 例如,如果你声明了这个...

前端老手
56分钟前
4
0
exist和in

exist和in select ..from table where exist (子查询) ; select ..from table where 字段 in (子查询) ; 如果主查询的数据集大,则使用In,效率高。 如果子查询的数据集大,则使用exist,效率高...

潦草的犀牛
今天
5
0
Android OkHttp + Retrofit 取消请求的方法

本文链接 前言 在某一个界面,用户发起了一个网络请求,因为某种原因用户在网络请求完成前离开了当前界面,比较好的做法是取消这个网络请求。对于OkHttp来说,具体是调用Call的cancel方法。 ...

shzwork
今天
6
0
并发编程之Callable异步,Future模式

Callable 在Java中,创建线程一般有两种方式,一种是继承Thread类,一种是实现Runnable接口。然而,这两种方式的缺点是在线程任务执行结束后,无法获取执行结果。我们一般只能采用共享变量或...

codeobj
今天
7
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部