C# windows服务中启动带UI界面的程序

本文详细介绍了如何在Windows服务中启动和关闭带有UI界面的外部程序,包括正常WinForm程序的启动方式以及在Windows服务中启动程序并绕过UAC的方法。此外,还提供了关闭程序的两种方式。文章讨论了在不同环境下启动程序可能遇到的问题,并给出了可能的解决方案。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1.正常winform程序中启动 外部程序:

               //打开程序的地址
                var fileName = path;
                //判断文件是否存在,如果不存在,返回 false
                Console.WriteLine(File.Exists(fileName));

                ProcessStartInfo startInfo = new ProcessStartInfo();
                startInfo.FileName = fileName;
                Process process = new Process();
                process.StartInfo = startInfo;
                process.Start();
                Log.Info("打开程序成功");

2. 在windows服务中需要使用特定的代码才能启动外部程序并显示程序的UI界面:

 public static bool StartProcessAndBypassUAC(String applicationName, String command, out PROCESS_INFORMATION procInfo)
        {
            // 登陆进程的id
            uint winlogonPid = 0;
            // 用户的访问令牌                   登陆进程的访问令牌      登陆进程的句柄
            IntPtr hUserTokenDup = IntPtr.Zero, hPToken = IntPtr.Zero, hProcess = IntPtr.Zero;
            // 进程信息
            procInfo = new PROCESS_INFORMATION();
            // obtain the currently active session id; every logged on user in the system has a unique session id
            //   获取当前活动会话id;系统中每个登录的用户都有一个唯一的会话id
            uint dwSessionId = WTSGetActiveConsoleSessionId();
            // obtain the process id of the winlogon process that is running within the currently active session
            //获取当前活动会话中运行的winlogon进程(身份认证)的进程id即用户登陆窗口进程id
            Process[] processes = Process.GetProcessesByName("winlogon");
            foreach (Process p in processes)
            {
                if ((uint)p.SessionId == dwSessionId)
                {
                    winlogonPid = (uint)p.Id;
                }
            }
            // obtain a handle to the winlogon process
            // 获取winlogon进程的句柄
            hProcess = OpenProcess(MAXIMUM_ALLOWED, false, winlogonPid);
            // obtain a handle to the access token of the winlogon process  
            // 获取winlogon进程访问令牌的句柄
            if (!OpenProcessToken(hProcess, TOKEN_DUPLICATE, ref hPToken))
            {
                // 使登陆进程句柄无效
                CloseHandle(hProcess);
                return false;
            }
            // Security attibute structure used in DuplicateTokenEx and CreateProcessAsUser  
            // I would prefer to not have to use a security attribute variable and to just   
            // simply pass null and inherit (by default) the security attributes  
            // of the existing token. However, in C# structures are value types and therefore  
            // cannot be assigned the null value.  
            // DuplicateTokenEx和CreateProcessAsUser中使用的安全属性结构我希望不必使用安全属性变量
            //只需传递null并继承(默认情况下)安全属性现有令牌的。然而,在C语言中,结构是值类型,因此无法分配空值。
            // 初始化安全属性结构体
            SECURITY_ATTRIBUTES sa = new SECURITY_ATTRIBUTES();
            sa.Length = Marshal.SizeOf(sa);
            // copy the access token of the winlogon process; the newly created token will be a primary token
            // 复制winlogon进程的访问令牌;新创建的令牌将是主令牌
            if (!DuplicateTokenEx(hPToken, MAXIMUM_ALLOWED, ref sa, (int)SECURITY_IMPERSONATION_LEVEL.SecurityIdentification, (int)TOKEN_TYPE.TokenPrimary, ref hUserTokenDup))
            {
                // 使登陆进程句柄无效
                CloseHandle(hProcess);
                // 使登陆进程的令牌句柄无效
                CloseHandle(hPToken);
                return false;
            }
            // By default CreateProcessAsUser creates a process on a non-interactive window station, meaning  
            // the window station has a desktop that is invisible and the process is incapable of receiving  
            // user input. To remedy this we set the lpDesktop parameter to indicate we want to enable user   
            // interaction with the new process.  
            // 默认情况下,CreateProcessAsUser在非交互式窗口站上创建进程,这意味着窗口站有一个不可见的桌面,进程无法接收用户输入。
            // 为了解决这个问题,我们设置了lpDesktop参数来表示我们想要启用用户与新流程的互动。
            // 用于指定新进程的主视窗特性的结构体
            STARTUPINFO si = new STARTUPINFO();
            si.cb = (int)Marshal.SizeOf(si);
            // 交互式窗口站参数;基本上,这表明创建的进程可以在桌面上显示GUI
            si.lpDesktop = @"winsta0\default"; // interactive window station parameter; basically this indicates that the process created can display a GUI on the desktop  
            // flags that specify the priority and creation method of the process  
            // 指定进程的优先级和创建方法的标志
            int dwCreationFlags = NORMAL_PRIORITY_CLASS | CREATE_NEW_CONSOLE;
            // create a new process in the current user's logon session  
            // 在当前用户的登录会话中创建新进程
            bool result = CreateProcessAsUser(hUserTokenDup,        // client's access token  用户的访问令牌
                                            applicationName,        // file to execute  要执行的文件
                                            command,                // command line  命令行
                                            ref sa,                 // pointer to process SECURITY_ATTRIBUTES 指向进程SECURITY_属性的指针 
                                            ref sa,                 // pointer to thread SECURITY_ATTRIBUTES  指向线程SECURITY_属性的指针
                                            false,                  // handles are not inheritable  句柄不可继承
                                            dwCreationFlags,        // creation flags  创建标志
                                            IntPtr.Zero,            // pointer to new environment block 指向新环境块的指针  
                                            null,                   // name of current directory  当前目录的名称 
                                            ref si,                 // pointer to STARTUPINFO structure  指向STARTUPINFO结构的指针
                                            out procInfo            // receives information about new process  接收有关新进程的信息
                                            );
            // invalidate the handles  使句柄无效
            CloseHandle(hProcess);
            CloseHandle(hPToken);
            CloseHandle(hUserTokenDup);
            return result; // return the result  
        }

接口调用:

StartProcessAndBypassUAC(@"D:\tes.exe", string.Empty, out pInfo);

备注:在我的电脑上能正常启动程序,但放到另外的电脑上不能启动,我怀疑是 si.lpDesktop = @"winsta0\default";这句代码的问题,如果没有这句代码,程序会以当前桌面为基础创建出UI程序进程,我没有测试过,如果你遇到这个问题可以修改下看看;

3. windows服务中关闭程序:

  1)通过获取进程列表数据,删除进程:

 Process[] processes = Process.GetProcesses();
            foreach (Process p in processes)
            {

                if (p.ProcessName == path)
                {
                    p.Kill();
                }

            }

2) 根据2中的pInfo结构体中的id删除:

 Process p = Process.GetProcessById((int)pInfo.dwProcessId);
                p.Kill();

<think>好的,我现在需要回答用户关于如何在Windows服务启动UI程序或配置交互式桌面权限的问题。首先,我需要回顾一下用户提供的引用内容,特别是引用[3],里面提到了一些方法,比如通过服务启动程序需要用户登录桌面后运行,涉及界面交互,程序启动主要有几种方式。这可能包括修改服务属性、使用任务计划程序或第三方工具。 接下来,我需要结合自己的知识来详细解释这些方法。首先,Windows服务默认在Session 0中运行,而用户界面程序在用户登录后的不同会话中运行,这导致服务无法直接启动UI程序。所以,用户需要配置服务以允许与桌面交互,或者使用其他方法来跨会话启动程序。 从引用[3]来看,可能的方法包括修改服务的“允许服务桌面交互”选项,但根据我的记忆,这个选项在Windows Vista及更高版本中已经失效,因为Session 0隔离。因此,可能需要其他方法,比如使用任务计划程序,在用户登录时触发启动UI程序,或者修改服务的登录账户为本地系统账户并勾选允许交互,但实际可能无效,需要更深入的解决方案。 另外,用户可能需要使用特定的API,比如CreateProcessAsUser,来在用户会话中启动进程。或者利用Windows提供的工具如PSExec来指定会话。此外,引用[3]提到的注册服务方式启动程序可能需要结合任务计划程序,或者使用SC命令修改服务配置,尽管可能受限。 我还需要确保回答结构清晰,分步骤说明每种方法,并指出注意事项,比如权限问题、安全风险等。同时,生成的相关问题需要围绕这个主题展开,比如不同Windows版本的区别、其他跨会话启动方法,或者如何以编程方式实现。 需要验证这些方法是否仍然适用,例如,在较新的Windows版本中,任务计划程序仍然是一个可靠的方法,而修改服务的交互式选项可能不再有效。同时,提到的WTSEnumerateSessions和CreateProcessAsUser可能需要管理员权限和正确的令牌处理,这对开发者来说可能比较复杂。 最后,确保回答符合系统级指令:使用正确的中文,数学公式用$...$,独立公式用$$...$$,但这里可能不涉及数学内容。引用用户提供的引用[3]在适当的位置,比如在提到服务启动方法时添加引用标识[^3]。</think>在Windows服务启动UI程序或配置交互式桌面权限,需要解决会话隔离问题(Session 0隔离机制)。以下是具体实现方法及注意事项: ### 一、基础配置方法 1. **修改服务属性(仅限早期Windows系统)** - 打开服务管理器(`services.msc`) - 找到目标服务 → 右键属性 → "登录"选项卡 - 勾选**允许服务桌面交互**(此选项在Windows Vista及更高版本已失效[^3]) 2. **通过任务计划程序启动** - 创建任务时勾选**运行用户登录时**,并设置触发器为用户登录事件 - 操作类型选择"启动程序",指定UI程序路径 - 安全选项选择**只在用户登录时运行** ### 二、进阶技术方案 ```csharp // 使用C#通过Windows API跨会话启动进程 [DllImport("advapi32.dll", SetLastError = true)] static extern bool CreateProcessAsUser( IntPtr hToken, string lpApplicationName, ...); ``` 需配合`WTSEnumerateSessions`获取用户会话令牌后执行。 ### 三、权限配置方案 1. **修改服务权限策略** ```bat sc config <服务名> type= interact type= own ``` 注意:Windows 10/11默认禁用此配置,需配合注册表修改: ``` HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\Control\Windows\NoInteractiveServices=0 ``` 2. **使用交互式服务检测(需重启)** - 组策略中启用**显示交互式服务检测通知** - 通过`Win32_Service`类设置`DesktopInteract`属性 ### 四、安全注意事项 - 跨会话操作需`SYSTEM`或`Administrator`权限 - 微软官方建议采用IPC通信替代直接启动UI - 需在程序清单中声明`uiAccess="true"`并签名
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值