EnvJS/Rhino, setTimeout() не работает
В настоящее время я установил EnvJS в моей системе (установлен здесь). Моя конечная цель - загрузить страницу, позволить ей в течение нескольких секунд обрабатывать javascript, а затем прочитать dom, чтобы получить интересующую информацию. Однако я не могу заставить setTimeout() работать, чтобы спасти мою жизнь (или JQuery в этом отношении).
У меня есть PHP-скрипт, который запускает процесс:
...
$ENVJS_PATH = "/var/www/project/src/envjs";
$RHINO_JAR = "rhino/js.jar";
$INIT_SCRIPT = "init.js";
$output = shell_exec("java -jar $ENVJS_PATH/$RHINO_JAR -opt -1 $ENVJS_PATH/$INIT_SCRIPT");
echo "Response from javascript:<br/> $output";
...
файл init.js выглядит так:
load('/var/www/project/src/envjs/dist/env.rhino.js');
print("Loaded env.rhino.js<br/>");
// temporarily commented out
//var url = "http://www.ken-soft.com";
//window.location = url;
//print("<br/>Loaded "+url+"<br/>");
// Problem starts here
var runAfterPause=function() {
print("got here..."); // never gets called
print(document.getElementById('some_id').innerHTML);
}
setTimeout(runAfterPause, 3000); //wait three seconds before continuing
// i have also tried setTimeout("runAfterPause()", 3000);
print("<br/>End<br/>");
Любые знания по этому вопросу будут высоко оценены. Благодарю.
4 ответа
Попробуйте env.rhino.1.2.js - а если серверная ОС, хостинг rhino
такое Ubuntu, тогда попробуй sudo apt-get install rhino
- и позвони rhino -opt -1 ...
вместо java -jar ...
Кажется, работает для меня на Ubuntu 11.04, когда запускается непосредственно на оболочке - не уверен, что PHP shell_exec
может влиять на вещи или нет..
РЕДАКТИРОВАТЬ: Действительно, это на самом деле не работает; Я немного просмотрел источник и увидел, что setTimeout
опирается на Timer.prototype.start = function(){};
который, видимо, пуст. Просмотр далее, единственное, что, кажется, имеет дело с синхронизацией Envjs.wait()
- и, используя это, я, наконец, могу получить своего рода синхронизированный цикл; однако, обратите внимание, что теперь он выглядит строго однопоточным (синхронным):
print("loading " + 1.2);
load('env.rhino.1.2.js'); // takes a while ...
print("loaded " + 1.2);
console.log(window);
var c=0;
function timedCount() // like this, when setTimeout calls a string!
{
c=c+1;
print("c=" + c);
if (c<10) // make a limit for the run of script:
{
var t;
//~ t=window.setTimeout(timedCount(),100); // TypeError: fn is not a function, it is undefined.
t=window.setTimeout("timedCount()",1000); // must have `t=...` - else it locks on return even w/ wait(0)!
Envjs.wait(); // waits, but "timer error undefined TypeError: fn is not a function, it is undefined." if setTimout doesn't call string; wait(0) exits immediately
} else Envjs.wait(0); // "reset": execute all timers and return; else here will be left hanging from previous wait()
}
// main:
timedCount();
//~ eval("timedCount()", null); // works the same
print("after timedCount()");
... и результаты:
$ sudo apt-get install rhino
$ wget https://github.com/thatcher/env-js
$ rhino -opt -1 test.js
loading 1.2
[ Envjs/1.6 (Rhino; U; Linux i386 2.6.38-11-generic; en-US; rv:1.7.0.rc2) Resig/20070309 PilotFish/1.2.13 ]
loaded 1.2
[Window]
a
c=1
c=2
c=3
c=4
c=5
c=6
c=7
c=8
c=9
c=10
after timedCount()
Если я правильно помню, в браузере setInterval
является асинхронным / многопоточным - действительно, в браузере JavaScript Shell 1.4 почти такой же код:
var c=0;
function timedCount()
{
c=c+1;
print("c=" + c);
if (c<10) {
var t;
t=window.setTimeout("timedCount()",1000);
}
}
timedCount();
print("after timedCount()");
производит:
c=1
after timedCount()
c=2
c=3
c=4
c=5
c=6
c=7
c=8
c=9
c=10
Метод обратного вызова определяется после назначения. Попробуйте поставить его перед setTimeout
Печать - это метод окна.
Он используется для печати страницы с помощью принтера...
Там может быть конфликт с ним и вашим методом печати.
Вот 2 вещи, которые я хочу попробовать для отладки. По крайней мере, визуально я не вижу что-то не так в вашем коде, если кто-то не видит это.
работает ли ваша функция, когда вы вызываете ее самостоятельно? не в установленном тайм-ауте
var runAfterPause=function() {
print("got here...");
print(document.getElementById('some_id').innerHTML);
}
// call function by it self
runAfterPause();
2-й попробуйте запустить его как анонимную функцию внутри вашего setTimeout;
var delay = setTimeout(function () {
print("got here..."); // never gets called
print(document.getElementById('some_id').innerHTML);
},3000);
это должно помочь вам отладить ваш код и увидеть, где находится ошибка.