Windows服务启动exe无界面终极解决方案

 

 

1、前言

我这个方案(C#操作)是彻底解决【从Windows服务启动程序exe,程序无界面】问题的终极解决方案,终极方案,绝对的终极方案。后附代码下载地址。

由于安全性问题,Vista以后的Windows都会出现该问题,从服务中调用/启动其他程序出现无界面,但是任务管理器中可以看到已经成功启动,就是无操作界面,具体出现该状况的原因大家自行搜索。我的方法绝对可行。

PS:服务是自己写的,要实现其他特殊功能,其他手法无法轻易绕过去。CSDN一点东西就要开VIP,那帮作者,呵呵,好东西还是在cnblogs里吧。

简而言之,本文的应用场景是:服务必须自建,还要打开外部exe

2、网上方案

网上有各种各样的方案,绝大部分都有一样的,都是是调用系统API,CreateProcess之类的API,然并卵,并不能彻底解决。

***2023-07-03****补充***

看到有人评价使用NSSM,我这服务是自己写的,因为它要实现其他特殊的功能,所以nssm什么的没有用哦,反而复杂了。

***2023-07-03****补充***

3、我的方案

我的方案极其简单而不粗暴,反而优美。

3.1、简单不粗暴


我的方案是使用计划任务功能启动指定程序。任务计划的启动不受服务限制,和服务的边界不太一样。不需要特别多的代码就可以实现,其实就是实现添加任务计划,简单吧。API方式,光结构和调试就够你们喝一壶了,还得通过其他API调用和设置其他信息,比如创建和复制现有执行令牌(DuplicateTokenEx方法),实现这功能粗暴得狠。我这个不用,啥都不用。

3.2、优美

创建任务寥寥十几行代码,优美得狠。

4、实现过程


实现过程即任务计划实现过程,C#有3种方法,其实就是2种,一种是使用API创建任务,这个方法其实,可以通过系统调用现有dll库实现,最后一种是使用开源库。建议用开源库方式。

调用系统的dll,这dll就是C:\Windows\System32\taskschd.dll,在C#里直接引用就行,它实现API的C#封装,很简单。使用TaskSchedulerClass类连接、创建修改任务计划,很简单,我这不是主推方法,不贴代码,但源码地址里有。

使用开源库TaskScheduler,可以实现,命名空间为Microsoft.Win32.TaskScheduler,下载地址为:https://github.com/dahall/TaskScheduler。例子为:https://github.com/dahall/TaskScheduler/wiki/Examples

    public static void AddOrRunWinTask( string sTaskName, string sExePath, string sArgs = null )
    {
        var task = TaskService.Instance.FindTask(sTaskName, true);

        if ( task != null )
        {
            task.Definition.Triggers[0].StartBoundary = DateTime.Now.AddSeconds ( 10 );
            task.RegisterChanges ();
        }
        else
        {
            var td = TaskService.Instance.NewTask ();

            td.RegistrationInfo.Author = "白羊佐CnBlog";
            td.RegistrationInfo.Description = "用于跨域启动特定程序";

            td.Settings.ExecutionTimeLimit = TimeSpan.Zero;//
            td.Settings.DisallowStartIfOnBatteries = false;
            td.Settings.RunOnlyIfIdle = false;
            td.Settings.RunOnlyIfNetworkAvailable = false;
            //此处注意,如果你待启动程序需要管理员权限运行,必须使用Highest,否则使用LUA就行
            td.Principal.RunLevel = TaskRunLevel.Highest;
            //获取Administrators的GroupID
            string sGpId = GetGroupID();
            //此处最为关键,如果不指定用户名ID或组名ID,依旧不显示界面,因为创建时的用户为SYSTEM
            td.Principal.GroupId = sGpId;

            var trigger = (TimeTrigger)td.Triggers.Add( new TimeTrigger() );
            trigger.StartBoundary = DateTime.Now.AddSeconds ( 10 );
            trigger.ExecutionTimeLimit = TimeSpan.Zero;
            trigger.Enabled = true;
            td.Actions.Add ( new ExecAction ( sExePath, sArgs ) );

            task = TaskService.Instance.RootFolder.RegisterTaskDefinition ( sTaskName, td );
        }

        //打开表示立即运行(切运行两次,因为上面有个执行延时)
        //var rz = task.Run ();
    }

   注意,注意,再注意:A、此方法是win10的,因为win10默认屏蔽Administrator用户,我用户属于这个组,所以我这个地方使用这种方式没有问题。但是,其他非Administrators组用户登录可能不行了,那也好解决,将下面代码中GroupPrincipal改为UserPrincipal,用它去找登录的用户名,上面代码设userid,logontype就行。或者输入正确的组名都可以;B、此方式程序的启动位置为System32,你程序的目录获取时要注意了。

    private static string GetGroupID ()
    {
        string sGid = null;
        
        System.DirectoryServices.AccountManagement.PrincipalContext pc = new System.DirectoryServices.AccountManagement.PrincipalContext(System.DirectoryServices.AccountManagement.ContextType.Machine);
        var identity = System.DirectoryServices.AccountManagement.GroupPrincipal.FindByIdentity(pc, "Administrators");
        
        if ( identity != null )
        {
            sGid = identity.Sid.Value;
        }

        return sGid;
    }

5、工具及代码下载地址

    https://files.cnblogs.com/files/ZoeWong/TaskScheduler.2.10.1%E5%8C%85.rar?t=1688102826&download=true

6、收尾

  哈哈,这个方法怎么样。彻底么?

 

posted @ 2023-06-30 13:37  白羊佐  阅读(1902)  评论(11编辑  收藏  举报