- xml - AJAX/Jquery XML 解析
- 具有多重继承的 XML 模式
- .net - 枚举序列化 Json 与 XML
- XML 简单类型、简单内容、复杂类型、复杂内容
使用 https://github.com/kriskowal/q图书馆,我想知道是否有可能做这样的事情:
// Module A
function moduleA_exportedFunction() {
return promiseReturningService().then(function(serviceResults) {
if (serviceResults.areGood) {
// We can continue with the rest of the promise chain
}
else {
performVerySpecificErrorHandling();
// We want to skip the rest of the promise chain
}
});
}
// Module B
moduleA_exportedFunction()
.then(moduleB_function)
.then(moduleB_anotherFunction)
.fail(function(reason) {
// Handle the reason in a general way which is ok for module B functions
})
.done()
;
基本上,如果服务结果不好,我想处理模块 A 中的故障,使用特定于模块 A 内部的逻辑,但仍然跳过 promise 链中剩余的模块 B 函数。
跳过模块 B 函数的明显解决方案是从模块 A 中抛出错误/原因。但是,然后我需要在模块 B 中处理它。理想情况下,我想在不需要任何额外代码的情况下执行此操作模块 B。
这很可能是不可能的 :) 或者违背 Q 的一些设计原则。
在这种情况下,您会建议哪种替代方案?
我有两种方法,但都有其缺点:
从模块 A 中抛出一个特定的错误并向模块 B 添加特定的处理代码:
.fail(function(reason) {
if (reason is specificError) {
performVerySpecificErrorHandling();
}
else {
// Handle the reason in a general way which is ok for module B functions
}
})
在模块A中进行自定义的错误处理,然后在处理完错误后,抛出一个伪造的拒绝原因。在模块B中,添加一个条件来忽略假的原因:
.fail(function(reason) {
if (reason is fakeReason) {
// Skip handling
}
else {
// Handle the reason in a general way which is ok for module B functions
}
})
方案一需要在模块B中添加模块A的具体代码。
解决方案 2 解决了这个问题,但整个伪造拒绝方法似乎很老套。
您能推荐其他解决方案吗?
最佳答案
在 JavaScript 中,调用函数时代码以两种方式流动。
返回
一个值给调用者,表明它成功完成。抛出
一个错误,表明发生了异常操作。 它看起来像:
function doSomething(){ // every function ever
if(somethingBad) throw new Error("Error operating");
return value; // successful completion.
}
try{
doSomething();
console.log("Success");
} catch (e){
console.log("Boo");
}
Promises 模拟这种完全相同的行为。
在 Promises 中,当您在 .then
处理程序中调用函数时,代码恰好以两种方式流动:
返回
一个 promise 或一个表明它成功完成的值。抛出
一个错误,表明发生了异常状态。它看起来像:
var doSomething = Promise.method(function(){
if(somethingBad) throw new Error("Error operating");
return someEventualValue(); // a direct value works here too
}); // See note, in Q you'd return Q.reject()
Promise.try(function(){ // in Q that's Q().then
doSomething();
console.log("Success");
}).catch(function(e){
console.log("Boo");
});
Promise 是对排序操作 本身概念的抽象。它描述了控制如何从一个语句传递到另一个语句。您可以将 .then
视为对分号的抽象。
让我们看看同步代码在您的情况下会是什么样子。
function moduleA_exportedFunction() {
var serviceResults = someSynchronousFunction();
if (serviceResults.areGood) {
// We can continue with the rest of our code
}
else {
performVerySpecificErrorHandling();
// We want to skip the rest of the chain
}
}
因此,如何继续我们的其余代码就是简单地返回
。这在同步代码和带有 promise 的异步代码中是一样的。执行非常具体的错误处理也是可以的。
我们如何跳过同步版本中的其余代码?
doA();
doB();
doC(); // make doD never execute and not throw an exception
doD();
好吧,即使不是立即,也有一种相当简单的方法可以让 doD 永远不会执行,方法是让 doC 进入无限循环:
function doC() {
if (!results.areGood) {
while(true){} // an infinite loop is the synchronous analogy of not continuing
// a promise chain.
}
}
因此,有可能永远无法解决 promise - 就像其他答案所暗示的那样 - 返回一个未决的 promise 。然而,这是非常糟糕的流量控制,因为意图没有很好地传达给消费者,并且可能很难调试。想象一下以下 API:
moduleA_exportedFunction - this function makes an API request and returns the service as a
ServiceData
object if the data is available. Otherwise, it enters the program into an endless loop.
有点令人困惑,是不是 :)?但是,它确实存在于某些地方。在非常古老的 API 中发现以下内容并不少见。
some_bad_c_api() - this function foos a bar, on failure it terminates the process.
那么,到底是什么困扰着我们终止该 API 中的进程?
在你的情况下。 ModelA 只是违反了它的责任限制,它不应该有权对程序的流程做出这样的决定。消费它的人应该做出这些决定。
更好的解决方案是抛出错误并让消费者处理。我将使用 Bluebird promises在这里,因为它们不仅快了两个数量级,而且具有更现代的 API - 它们还具有更好的调试工具 - 在这种情况下 - 用于条件捕获和更好的堆栈跟踪的糖:
moduleA_exportedFunction().then(function(result){
// this will only be reached if no error occured
return someOtherApiCall();
}).then(function(result2){
// this will be called if the above function returned a value that is not a
// rejected promise, you can keep processing here
}).catch(ApiError,function(e){
// an error that is instanceof ApiError will reach here, you can handler all API
// errors from the above `then`s in here. Subclass errors
}).catch(NetworkError,function(e){
// here, let's handle network errors and not `ApiError`s, since we want to handle
// those differently
}).then(function(){
// here we recovered, code that went into an ApiError or NetworkError (assuming
// those catch handlers did not throw) will reach this point.
// Other errors will _still_ not run, we recovered successfully
}).then(function(){
throw new Error(); // unless we explicitly add a `.catch` with no type or with
// an `Error` type, no code in this chain will run anyway.
});
所以在一行中 - 你会做你会在同步代码中做的事情,就像 promises 通常的情况一样。
注意 Promise.method 只是 Bluebird 用于包装函数的便利函数,我只是讨厌同步投入 promise 返回 API,因为它会造成重大破坏。
关于javascript - 处理错误后跳过 promise 链,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/24355960/
我的算法- private static MyList skip$DeleteItem(MyList L , int M , int N){ MyList curr = L; MyLi
我正在 SWI-Prolog 下开发,但我的目标是 Erlog (https://github.com/rvirding/erlog)。我需要一种使用非标准 Prolog 语法的方法。 有没有办法
我正在尝试从应用程序下载一大堆文件。它的shell命令是“下载文件名”。 我有一个文本文件,其中包含必须下载的所有文件名。我要做的就是运行一个脚本/命令,以便在执行上述命令时 1.从文本文件中提取文件
我试图循环遍历所有用户的评论,但使用 if 语句查找特定值。问题是我的应用程序崩溃了,因为一些用户没有发表评论,因此我得到“无法读取‘收集’未定义的属性”。如何跳过 if 语句的未定义值?代码如下:
我们有按年份分区的索引,例如: items-2019 items-2020 考虑以下数据: POST items-2019/_doc { "@timestamp": "2019-01-01" }
我只是编写一个页面来按实体编号查看每个 ASCII 条目,我想知道是否有一种更简单/更干净的方法来跳过不需要的数字。 var x = new Ar
我希望能够普遍使用重复条目,但也能够跳过特定日期。例子: ** TODO swim practice SCHEDULED 但是,我提前知道 2013-12-25 不会有练习。但是,当我将项目标
如何跳过像这样的 for 循环的一次迭代: for (int i = 65; i <= 90; i++) { if (!(i == 73)) { uniq.add((char) i);
这个问题已经存在: Scanner issue when using nextLine after nextXXX [duplicate] 已关闭 9 年前。 ask=1; while(ask==1)
我在使用一个程序时遇到了一些麻烦,我应该允许用户在程序中输入任意数量的数字,直到他们不再想要为止。然后程序应该计算输入数字的平均值和最大值。我哪里做错了? import java.util.Scann
我有一个名为segments的 Sprite 数组,我想在每个循环中跳过segments的第一个元素。我目前正在这样做: var first = true; for each (var segment
我目前正在编写一个 for 循环来遍历包含 38 个元素的 2D。然而,其中一些元素为空,我希望 for 循环简单地跳过它们(因为在我正在解决的难题中,它们没有与它们相关的移动)。快速搜索表明,我可以
我想使用pre-commit处理我的 git 项目的 git hooks。但是,当我使用它时,git commit 命令不断跳过 unittest 执行: (smartexchange) trnboo
这个问题在这里已经有了答案: Does scanf() take '\n' as input leftover from previous scanf()? (1 个回答) 关闭 9 年前。 我正在
我正在迭代 csv 文件中的每一行,并仅选择/计算满足条件的行。但是,当连续出现错误时,它会停止循环。有没有办法告诉 python 跳过错误并移动到下一行?我使用了 try 函数但没有工作。我的代码是
感谢您提供的优秀示例,我试过了,它按我的预期工作。很高兴看到有人了解问题的本质。但是,我认为我应该用 Lift 标记问题,因为我正在使用 Lift 框架,这就是(仍然)发生这个问题的地方(尽管我仍然认
大家好,我正在编写一个算法来从 NodeTree 中删除具体分支(例如 DSF)。如果您选择 Node 的名称,算法将检查该 Node 是否是其他 Node 的父 Node ;如果是,它会获取该 No
附有代码和输出。 基本上它是第二次跳过我的输入。就像我启动代码一样,它让我输入一个选项,然后第二次跳过输入,直接转到开关的默认情况。 然后第三次它就会允许我输入。不明白为什么。 任何帮助将不胜感激。
我在 JavaScript 中有一个 for 循环,它会定期跳过间隔,但我无法确定原因。 console.log(parseInt($('input.num-to-add').val())); num
我正在 JasperSoft 中填写参数。在我的报告中我有参数:参数_1、参数_2、参数_3 int a; for (a = 0; a < headers.length; a++) {
我是一名优秀的程序员,十分优秀!