我想在循环等待选择器中做,只要找到的对象不满足指定的条件。我写了下面的代码。
casper.then(
function(){
var need_exit = false;
this.echo('enter loop');
var i = 1;
while (!need_exit){
this.echo('check exit');
// check, if chat was finished
if (this.exists('div#chat > p')){
need_exit = true;
this.echo('exit loop');
}
// wait new message
else {
this.echo('wait msg ' + datetime());
var selector = 'div#chat tr:nth-child('+i+')';
try{
this.waitForSelector(selector,
function(){
var msg_selector = 'div#chat tr:nth-child('+i+') > td:nth-child(2)';
var inf = this.getElementInfo(msg_selector);
this.echo(inf['text']);
i++;
},
null,
5000);
}
catch(err){
this.echo('[wait timeout]');
}
need_exit = true;
}
}
}
);
问题是迭代不断跟随,而不是在找到项目或超时到期后前进到下一次迭代。为什么会发生这种情况以及如何做到正确?
答案 0 :(得分:1)
问题是所有then*
和wait*
调用都是异步步骤函数。这就是为什么你不能绕过它们的原因。解决这个问题的正常方法是使用递归函数:
这是您功能的重新设想版本:
casper.waitContinuouslyUntilSelector = function(checkRow, finalSelector, then, onTimeout, timeout, i){
// TODO: remove the `this.echo()` calls from this function
i = i || 1;
this.then(function(){
if (this.exists(finalSelector)) {
this.echo('finalSelector found');
this.then(then);
return;
}
this.waitForSelector(checkRow(i), function _then(){
this.waitContinuouslyUntilSelector(checkRow, finalSelector, then, onTimeout, timeout, i+1);
}, function _onTimeout(){
if (this.exists(finalSelector)) {
this.echo('finalSelector found');
this.then(then);
} else {
this.echo('finalSelector not found');
if (typeof onTimeout !== "function") {
throw new CasperError("Final selector was not found and next row was not loaded");
}
this.then(onTimeout);
}
}, timeout);
});
return this;
}
你可以像这样使用它:
casper.then(function(){
var bindingExitSelector = 'div#chat > p';
var rowSelectorFunc = function(i){
return 'div#chat tr:nth-child('+i+') > td:nth-child(2)';
};
this.waitContinuouslyUntilSelector(rowSelectorFunc, bindingExitSelector);
});
如果没有更多的行加载,那么&#34; final&#34; <p>
也不存在,出现超时错误。如果您想阻止这种情况,则需要传入onTimeout
回调:
this.waitContinuouslyUntilSelector(rowSelectorFunc,
bindingExitSelector,
null,
function _onTimeout(){
this.echo("Failed to load the next row and didn't found the final selector; continue...");
});