【Win10 应用开发】从前台应用触发后台任务

原文:【Win10 应用开发】从前台应用触发后台任务

有关后台任务的使用,估计大伙伴们不会陌生,而且老周曾经在某文中也简单讲述过。说到后台任务,老周想到了一个问题:有人问,后台任务一定要独立写到一个Runtime组件中吗,能不能写到主项目的代码中?

老周严重地回答你:是可以的,在配置清单文件中,你只需要在Extension元素中指定Executable为主项目的.exe文件即可。

        <Extension Category="windows.backgroundTasks" EntryPoint="……" Executable="xxxx.exe">
          <BackgroundTasks>
            ……
          </BackgroundTasks>
        </Extension>

其实,Executable可以设置为$targetnametoken$.exe,这样在生成应用时,会自动用.exe文件的实际名称替换$targetnametoken$标签。

可是,老周非常严重地不推荐把后台任务写到主项目中,一则这不符合官方文档的常规性要求,不便于代码分类管理;二则,这种做法会带来严重的负面效果,如果后台任务激活时,前台应用没有运行倒无所谓,要是后台任务执行时前台应用正在运行,很可能会造成前台应用进程重启,给用户的感觉就是闪退。

 

好,上面只是给大家普及一下常识,建议大家在干活时最好按规矩办事,别老喜欢搞那些另类行为,另类不代表创新,只不过是幼稚罢了。

今天的主题是:我能不能在前台应用中,通过代码有目的地触发后任务呢?我们知道,常见的后台任务有后台音频、定时器、系统事件、网络传输控制等触发器,那有没有可以让我们手动去触发后台任务的触发器呢?

有,它的名字叫ApplicationTrigger,使用它,你在前台代码中就可以随时激活后台任务,而且还可以向后台任务传递参数。当需要执行后台任务时,直接调用ApplicationTrigger实例的RequestAsync方法即可,如果没有意外(比如泥石流、山体滑坡、地震等),那么后台任务就会执行。

 

原理已经跟大家交代了,下面还是用实例说话吧。

首先我定义了这么个后台任务,你猜猜它是干吗用的。

 

    public sealed class BgTask : IBackgroundTask
    {
        public async void Run(IBackgroundTaskInstance taskInstance)
        {
            var d = taskInstance.GetDeferral();
            // 取出与触发器相关的数据
            ApplicationTriggerDetails details = taskInstance.TriggerDetails as ApplicationTriggerDetails;

            if (details != null)
            {
                // 取出传递过来的参数
                ValueSet ps = details.Arguments;
                int na = Convert.ToInt32(ps["a"]);
                int nb = Convert.ToInt32(ps["b"]);

                // 开始进行运算
                int x = na;
                int res = 0;
                while (x <= nb)
                {
                    res += x;
                    x++;
                    await Task.Delay(30); //延时
                    // 报告进度
                    ReportProgress(taskInstance, (uint)x, (uint)nb);
                }

                // 保存计算结果
                ApplicationData.Current.LocalSettings.Values["result"] = res;
            }
            d.Complete();
        }

        private void ReportProgress(IBackgroundTaskInstance instance, uint c, uint t)
        {
            ……
        }
    }

人品高尚的你一定看出来了,这个任务主要是做加法运算的,指定一个起始值和一个最终值,从起始值开始相加,一直加到最终值,每加一次就把操作数+1。其中,每一轮运算后会拖延30毫秒,目的是方便看进度。本任务不太严谨,比如它没有检测如果终值小于始值时怎么处理,大家知道就行了,老周不想把代码搞复杂,仅供演示。

然后你必须记得在主项目中引用写有后台任务类的Rutime组件项目。

接下来,配置一下清单文件,打开Package.appxmanifest文件,找到Application节点,注意不是Applications节点,没有s的,随后加入扩展点。

    <Application Id="App" …… >
      ……
      <Extensions>
        <Extension Category="windows.backgroundTasks" EntryPoint="BackgroundTasks.BgTask" >
          <BackgroundTasks>
            <Task Type="general"/>
          </BackgroundTasks>
        </Extension>
      </Extensions>
    </Application>

<Task Type="general"/>表示该后任务是常规型,通用型,即大众化的后台任务,而不是精英级后台。

 

清单文件只是作声明,后台任务不会自动注册,需要用代码来完成注册。

        const string TASK_NAME = "comptask";
        private BackgroundTaskRegistration taskReg = null;
        protected override async void OnNavigatedTo(NavigationEventArgs e)
        {
            var res = await BackgroundExecutionManager.RequestAccessAsync();
            if (res == BackgroundAccessStatus.Unspecified || res == BackgroundAccessStatus.Unspecified)
            {
                ShowMessage("后台任务被禁用。"); return;
            }

            taskReg = BackgroundTaskRegistration.AllTasks.Values.FirstOrDefault(t => t.Name == TASK_NAME) as BackgroundTaskRegistration;
            // 注册后台任务
            if (taskReg == null)
            {
                BackgroundTaskBuilder bd = new BackgroundTaskBuilder();
                // 入口点
                bd.TaskEntryPoint = typeof(BackgroundTasks.BgTask).FullName;
                // 任务名称
                bd.Name = TASK_NAME;
                // 设置触发器
                ApplicationTrigger trigger = new ApplicationTrigger();
                bd.SetTrigger(trigger);
                taskReg = bd.Register();
                ShowMessage("后台任务注册成功。");
            }
            // 添加事件处理
            taskReg.Progress += TaskReg_Progress;
            taskReg.Completed += TaskReg_Completed;
        }

注册前应该访问BackgroundTaskRegistration.AllTasks看看你需要的后台任务是不是已经注册了,重复注册没有意义。

下面代码通过ApplicationTrigger来触发后台任务。

            // 从注册的任务中取出触发器
            ApplicationTrigger trigger = taskReg.Trigger as ApplicationTrigger;

            // 准备参数
            ValueSet p = new ValueSet();
            p["a"] = n1;
            p["b"] = n2;
            // 触发后台任务
            var res = await trigger.RequestAsync(p);
            switch (res)
            {
                case ApplicationTriggerResult.Allowed:
                    ShowMessage("后台任务已启动。");
                    break;
                case ApplicationTriggerResult.CurrentlyRunning:
                    ShowMessage("后台任务已经在运行了。");
                    break;
                case ApplicationTriggerResult.DisabledByPolicy:
                    ShowMessage("管理员不允许执行后台任务。");
                    break;
                case ApplicationTriggerResult.UnknownError:
                    ShowMessage("发生错误。");
                    break;
           }

 在调用RequestAsync方法时,可以向后台任务传递数据,数据用ValueSet类封装,其实就是个字典模型,key为字符串。

 

方法调用后,会返回ApplicationTriggerResult枚举的值。如果值为Allowed表明后台任务已成功触发;如果为CurrentlyRunning表明后台任务还在运行中,“你所拨打的号码正在通话中,请稍后再试”;如果值为DisabledByPolicy表示后台任务被禁止了。

 

一切就绪,看看运行结果:

 

呈现计算结果。

 

本示例下载地址:http://files.cnblogs.com/files/tcjiaan/triggerbgtfromapp.zip

 

好了,肚子饿了,要开饭,今天的白菜猪肉汤比较nice。有网友提出,老周能不能写一写Win10中Toast通知和操作中心的一些新知识点。没问题的,下一篇烂文开始,老周就和大伙伴们一起研究一下Adaptive Generic Toast吧。

 

时间: 2024-10-30 12:14:57

【Win10 应用开发】从前台应用触发后台任务的相关文章

Win10开发教程 如何从前台应用触发后台任务

本文我们主要讲的是有关后台任务的使用,估计大伙伴们不会陌生,后台任务一定要独立写到一个Runtime组件中吗,能不能写到主项目的代码中? 是可以的,在配置清单文件中,你只需要在Extension元素中指定Executable为主项目的.exe文件即可. <Extension Category="windows.backgroundTasks" EntryPoint="--" Executable="xxxx.exe">   <B

【Win10应用开发】自定义磁贴通知的排版

原文:[Win10应用开发]自定义磁贴通知的排版 前面老周用了两篇烂文,向大家介绍了Adaptive磁贴的模板使用.那些XML模板已经很强大了,不过,如果你觉得那些排版还不足以满足需求,不妨试试自己来定义磁贴的内容. 其实,Runtime App支持在后台任务中生成XAML呈现,只要你编写的后台任务类从XamlRenderingBackgroundTask(位于Windows.UI.Xaml.Media.Imaging命名空间)类派生即可.利用这一特性,我们可以在后台生成XAML布局,然后通过R

Win10 UWP 开发系列:使用SplitView实现汉堡菜单及页面内导航

原文:Win10 UWP 开发系列:使用SplitView实现汉堡菜单及页面内导航 在Win10之前,WP平台的App主要有枢轴和全景两种导航模式,我个人更喜欢Pivot即枢轴模式,可以左右切换,非常方便.全景视图因为对设计要求比较高,自己总是做不出好的效果.对于一般的新闻阅读类App来说,Pivot更适合多个频道的展示,因为内容基本都是一样的. 到了Win10,微软模仿其他平台也推出了汉堡菜单,但并没有提供现成的控件,而是需要开发者通过一个名为SplitView的控件来实现.我个人并不觉得左上

“Win10 UAP 开发系列”之 在MVVM模式中控制ListView滚动位置

原文:"Win10 UAP 开发系列"之 在MVVM模式中控制ListView滚动位置 这个扩展属性从WP8.1就开始用了,主要是为了解决MVVM模式中无法直接控制ListView滚动位置的问题.比如在VM中刷新了数据,需要将View中的ListView滚动到顶部,ListView只有一个ScrollIntoView()方法可以控制滚动的位置,但最好在VM中不要出现直接控制View的代码,需要通过其他的方式. 使用一个扩展属性即可实现: /// <summary> ///

Win10 UWP开发系列:实现Master/Detail布局

原文:Win10 UWP开发系列:实现Master/Detail布局 在开发XX新闻的过程中,UI部分使用了Master/Detail(大纲/细节)布局样式.Win10系统中的邮件App就是这种样式,左侧一个列表,右侧是详情页面.关于这种 样式的说明可参看MSDN文档:https://msdn.microsoft.com/zh-cn/library/windows/apps/xaml/dn997765.aspx 样式如下: 在微软官方的Sample里,有这种样式的代码示例,下载地址:https:

【Win10 应用开发】集成语音命令

原文:[Win10 应用开发]集成语音命令 记得老周以前在写WP8应用开发的文章时,曾经写过语音命令集成的文章,后来8.1的时候"小娜"问世,但考虑到其变化不大,故老周没有补写相应的文章. 今天,老周打算补一下Win 10通用应用开发中,有关语音命令集成相关的内容.虽然还是一脉相承,大的变化没有,不过Win10 sdk在语音命令定义文件中添加了新内容,而且现在不仅能在手机应用中加入语音集成,在面向PC和板子的应用中也能如愿,因为应用程序已经通用. 同理,在开始之前,老周仍然先给大家讲个

【Win10应用开发】自定义打印选项

原文:[Win10应用开发]自定义打印选项 老周在前一篇烂文中已经给大伙伴们演示了如何打印UI元素,今天的烂文就向各位介绍一下,如何向打印对话框添加自定义选项.如果只是讲如何实现,会比较抽象,也比较枯燥,而且相当无聊,更是说不清楚,毕竟这打印API用起来要比其他API稍稍复杂了一点.所以老周就做了一个打印图片的垃圾应用,在打印对话框中,你可以选择设置要打印图片的不透明度.旋转角度(0度,90度,180度,270度). OK,为环保事业做贡献,老周今天也节约一点口水.下面咱们开始干活. 1.先来设

Win10 UWP 开发系列:使用多语言工具包让应用支持多语言

原文:Win10 UWP 开发系列:使用多语言工具包让应用支持多语言 之前我在一篇blog中写过如何使用多语言工具包,见http://www.cnblogs.com/yanxiaodi/p/3800767.html 在WinEcos社区也发布过一篇详细的文章介绍多语言工具包的使用,但因社区改版那篇文章已经找不到了. 当时写的时候还没有出Win10的SDK,都是基于UAP框架写的.微软早已经发布了Win10的SDK,相应的项目结构也发生了变化,以前分为两个项目通过Share项目共享代码的方式被抛弃

Win10 UWP开发系列:使用VS2015 Update2+ionic开发第一个Cordova App

原文:Win10 UWP开发系列:使用VS2015 Update2+ionic开发第一个Cordova App 安装VS2015 Update2的过程是非常曲折的.还好经过不懈的努力,终于折腾成功了. 如果开发Cordova项目的话,推荐大家用一下ionic这个框架,效果还不错.对于Cordova.PhoneGap.ionic.AngularJS这些框架或库的关系,我个人理解是这样,PhoneGap是一个商业项目,用来实现HTML5式的跨平台开发,后来Adobe公司将其中的核心代码开源,就是Co