编程语言
首页 > 编程语言> > javascript – WinJS,从一个函数返回一个promise,它可能是也可能不是async

javascript – WinJS,从一个函数返回一个promise,它可能是也可能不是async

作者:互联网

我有一种情况,我的WinJS应用程序想要调用一个可能是也可能不是异步的函数(例如,在一种情况下我需要从文件中加载一些数据(异步),但在其他时候我可以同步加载缓存).

浏览文档我虽然可以将条件逻辑包装在一个承诺中:

A)
return new WinJS.Promise(function() {  // mystuff });

或者可能像这样使用’as’:

B)
return WinJS.Promise.as(function() { // mystuff });

问题是,当我调用这个函数时,我正在从我的第一页的ready()函数执行这样的操作:

WinJS.UI.Pages.define("/pages/home/home.html", {

    ready: function () {

         Data.Survey.init().done(function (result) {

            // do some stuff with 'result'

        });
    }
 });

当它写成’A’时,它永远不会打到我的done()调用.

或者如果我把它称为’B’,它会立即执行我的done()中的代码,然后才能解决.它还从结果的值看,它刚刚被设置为我的init()函数的内容,而不是被包含在一个promise中.

感觉我在这里做的事情基本上是错误的,但我不确定从哪里开始寻找.

如果有任何帮助,这是我的init()函数的精简版:

    function init() {

    return new WinJS.Promise(function() {

        if (app.context.isFirstRun) {

            app.surveyController.initialiseSurveysAsync().then(function (result) {
                return new WinJS.Binding.List(result.surveys);
            });
        } else {

            var data = app.surveyController.getSurveys();
            return new WinJS.Binding.List(data);
        }
    });
}  

有没有人对这个有任何想法?我不相信’可能或可能不是异步’是这里的问题,我相信承诺设置不是我所期望的.谁能在这里看到任何明显错误的东西任何反馈都非常感谢.

解决方法:

一般来说,如果您在完整的init例程中执行文件I / O,那些API本身会返回promise,在这种情况下,您希望从其中一个.then方法返回其中一个promise或promise.

另一方面,WinJS.Promise.as旨在将一个值包装在一个promise中.但让我更充分地解释一下.

首先,仔细阅读WinJS.Promise构造函数的文档.像许多其他人一样,你错误地认为你只是将一段代码包装在诺言中并且瞧!它是异步的.不是这种情况.传递给构造函数的函数是一个初始化程序,它接收三个参数:一个completeDispatcher函数,一个errorDispatcher函数和一个progressDispatcher函数,我喜欢调用它们.

对于永远完成成功,承诺错误或报告进度的承诺,初始化程序中的其余代码最终必须调用其中一个调度程序.这些调度程序在promise中,然后遍历并调用任何已完成/错误/进度方法,这些方法已经被赋予了该promise或者已完成的方法.因此,如果您根本没有调用调度程序,则无法完成,这正是您所看到的行为.

使用WinJS.Promise.as类似于它在promise中包含一个值.在您的情况下,如果您将一个函数传递给WinJS.promise.as,那么您将得到的是一个承诺,因此该函数值已经实现了.您没有获得该函数的异步执行.

要实现异步行为,您必须使用setTimeout / setInterval(或Windows 8.1中的WinJS调度程序)在UI线程上执行异步工作,或者使用Web worker作为后台线程并将其完成(通过postMessage)绑定到promise中.

这是使用构造函数创建promise的完整示例,处理完整,错误和进度情况(以及取消):

    function calculateIntegerSum(max, step) {
    if (max < 1 || step < 1) {
        var err = new WinJS.ErrorFromName("calculateIntegerSum", "max and step must be 1 or greater");
        return WinJS.Promise.wrapError(err);
    }

    var _cancel = false;

    //The WinJS.Promise constructor's argument is a function that receives 
    //dispatchers for completed, error, and progress cases.
    return new WinJS.Promise(function (completeDispatch, errorDispatch, progressDispatch) {
        var sum = 0;

        function iterate(args) {
            for (var i = args.start; i < args.end; i++) {
                sum += i;
            };

            //If for some reason there was an error, create the error with WinJS.ErrorFromName
            //and pass to errorDispatch
            if (false /* replace with any necessary error check -- we don’t have any here */) {
                errorDispatch(new WinJS.ErrorFromName("calculateIntegerSum", "error occurred"));
            }

            if (i >= max) {
                //Complete--dispatch results to completed handlers
                completeDispatch(sum);
            } else {
                //Dispatch intermediate results to progress handlers
                progressDispatch(sum);

                //Interrupt the operation if canceled
                if (!_cancel) {
                    setImmediate(iterate, { start: args.end, end: Math.min(args.end + step, max) });
                }
            }
        }

        setImmediate(iterate, { start: 0, end: Math.min(step, max) });
    },
    //Cancellation function 
    function () {
        _cancel = true;
    });
}

这来自我的免费电子书的附录A(“揭秘承诺”),用HTML,CSS和JavaScript编写Windows应用商店应用程序,第二版(预览版),见http://aka.ms/BrockschmidtBook2.

在您的情况下,您可以将数据初始化代码放在iterate函数的位置,也可以在setImmediate中调用它.我还建议您查看WinJS调度程序API,它可以让您在UI线程上设置工作的优先级.

简而言之,理解新的WinJS.Promise和WinJS.Promise.as本身并不会创建异步行为是至关重要的,因为promises本身只是围绕“稍后交付的结果”的调用约定,它与async没有任何内在联系. .

标签:winjs,javascript,asynchronous,promise
来源: https://codeday.me/bug/20190728/1564399.html