我试图在网页上运行一个进程,该进程将实时返回其输出。例如,如果我运行“ ping”进程,则每次返回新行时都应更新页面(现在,当我使用exec(command,output)时,我被迫使用-c选项,等到进程完成后才能看到在我的网页上输出)。是否可以在php中做到这一点?
我也想知道当有人离开页面时杀死这种过程的正确方法是什么。在“ ping”进程的情况下,我仍然能够在系统监视器中看到正在运行的进程(有意义)。
我试图在网页上运行一个进程,该进程将实时返回其输出。例如,如果我运行“ ping”进程,则每次返回新行时都应更新页面(现在,当我使用exec(command,output)时,我被迫使用-c选项,等到进程完成后才能看到在我的网页上输出)。是否可以在php中做到这一点?
我也想知道当有人离开页面时杀死这种过程的正确方法是什么。在“ ping”进程的情况下,我仍然能够在系统监视器中看到正在运行的进程(有意义)。
Answers:
这对我有用:
$cmd = "ping 127.0.0.1";
$descriptorspec = array(
0 => array("pipe", "r"), // stdin is a pipe that the child will read from
1 => array("pipe", "w"), // stdout is a pipe that the child will write to
2 => array("pipe", "w") // stderr is a pipe that the child will write to
);
flush();
$process = proc_open($cmd, $descriptorspec, $pipes, realpath('./'), array());
echo "<pre>";
if (is_resource($process)) {
while ($s = fgets($pipes[1])) {
print $s;
flush();
}
}
echo "</pre>";
ob_implicit_flush(true);ob_end_flush();
shell_exec
。如果您只想显示输出而不对其进行任何处理,则效果很好。
这是显示shell命令实时输出的好方法:
<?php
header("Content-type: text/plain");
// tell php to automatically flush after every output
// including lines of output produced by shell commands
disable_ob();
$command = 'rsync -avz /your/directory1 /your/directory2';
system($command);
您将需要此功能来防止输出缓冲:
function disable_ob() {
// Turn off output buffering
ini_set('output_buffering', 'off');
// Turn off PHP output compression
ini_set('zlib.output_compression', false);
// Implicitly flush the buffer(s)
ini_set('implicit_flush', true);
ob_implicit_flush(true);
// Clear, and turn off output buffering
while (ob_get_level() > 0) {
// Get the curent level
$level = ob_get_level();
// End the buffering
ob_end_clean();
// If the current level has not changed, abort
if (ob_get_level() == $level) break;
}
// Disable apache output buffering/compression
if (function_exists('apache_setenv')) {
apache_setenv('no-gzip', '1');
apache_setenv('dont-vary', '1');
}
}
它不适用于我尝试过的每台服务器,但我希望我能提供有关在php配置中查找内容的建议,以确定是否应该尝试使这种行为起作用在您的服务器上!还有其他人知道吗?
这是纯PHP中的一个虚拟示例:
<?php
header("Content-type: text/plain");
disable_ob();
for($i=0;$i<10;$i++)
{
echo $i . "\n";
usleep(300000);
}
我希望这对其他在这里搜索过的人有所帮助。
exec()
。system()
虽然工作正常。
header('X-Content-Type-Options: nosniff');
for ($i=0; $i < 100000; $i++) { echo " "; } echo "\n"; flush();
此处介绍了使用现代HTML5服务器端事件针对此旧问题的更好解决方案:
http://www.w3schools.com/html/html5_serversentevents.asp
例:
http://sink.agiletoolkit.org/realtime/console
代码:https://github.com/atk4/sink/blob/master/admin/page/realtime/console.php#L40
(作为模块在敏捷工具包框架中实现)
对于命令行用法:
function execute($cmd) {
$proc = proc_open($cmd, [['pipe','r'],['pipe','w'],['pipe','w']], $pipes);
while(($line = fgets($pipes[1])) !== false) {
fwrite(STDOUT,$line);
}
while(($line = fgets($pipes[2])) !== false) {
fwrite(STDERR,$line);
}
fclose($pipes[0]);
fclose($pipes[1]);
fclose($pipes[2]);
return proc_close($proc);
}
如果您尝试运行文件,则可能需要先授予其执行权限:
chmod('/path/to/script',0755);
试试看(在Windows计算机+ Wamp服务器上测试)
header('Content-Encoding: none;');
set_time_limit(0);
$handle = popen("<<< Your Shell Command >>>", "r");
if (ob_get_level() == 0)
ob_start();
while(!feof($handle)) {
$buffer = fgets($handle);
$buffer = trim(htmlspecialchars($buffer));
echo $buffer . "<br />";
echo str_pad('', 4096);
ob_flush();
flush();
sleep(1);
}
pclose($handle);
ob_end_flush();
我在Windows上尝试了各种PHP执行命令,发现它们之间有很大的不同。
shell_exec
,exec
,passthru
proc_open
,popen
-之类的,因为您无法将参数传递给命令(即,不会与一起使用my.exe --something
,将与一起使用_my_something.bat
)。最好的(最简单的)方法是:
header('Content-Type: text/event-stream');
(而不是header('Content-Type: text/plain; charset=...');
)。但是,这并非在所有浏览器/客户端中都适用!流媒体将在没有此功能的情况下工作,但是浏览器至少会缓冲第一行。header('Cache-Control: no-cache');
。ini_set('output_buffering', 'off');
)。这可能还必须在Apache / Nginx /您前面使用的任何服务器中完成。ini_set('zlib.output_compression', false);
)。这可能还必须在Apache / Nginx /您前面使用的任何服务器中完成。因此,在您的C ++程序中,您可以执行以下操作(再次,有关其他解决方案,请参见printf刷新问题):
Logger::log(...) {
printf (text);
fflush(stdout);
}
在PHP中,您可以执行以下操作:
function setupStreaming() {
// Turn off output buffering
ini_set('output_buffering', 'off');
// Turn off PHP output compression
ini_set('zlib.output_compression', false);
// Disable Apache output buffering/compression
if (function_exists('apache_setenv')) {
apache_setenv('no-gzip', '1');
apache_setenv('dont-vary', '1');
}
}
function runStreamingCommand($cmd){
echo "\nrunning $cmd\n";
system($cmd);
}
...
setupStreaming();
runStreamingCommand($cmd);
首先检查一下flush()是否适合您。如果可以,则很好,如果不能,则可能意味着Web服务器由于某种原因正在缓冲,例如已启用mod_gzip。
对于类似ping的东西,最简单的技术是在PHP中循环,多次运行“ ping -c 1”,并在每次输出后调用flush()。假设将PHP配置为在用户关闭HTTP连接时中止(通常是默认设置,或者您可以调用ignore_user_abort(false)来确保),那么您无需担心ping操作失控。
如果确实有必要只运行一次子进程并连续显示其输出,那可能会更困难-您可能必须在后台运行它,将输出重定向到流,然后让PHP回显该流返回用户,并穿插常规的flush()调用。
如果您希望通过PHP运行系统命令,请参阅exec文档。
我不建议在流量高的站点上执行此操作,因为为每个请求分叉一个过程是相当繁重的过程。某些程序提供将其进程ID写入文件的选项,以便您可以随意检查并终止该进程,但是对于ping之类的命令,我不确定是否可以检查手册页。
只需在远程主机上您希望侦听的端口(IE:HTTP的端口80)上打开一个套接字,便可以更好地为您服务,这样您就可以知道用户区以及网络上一切正常。
如果您尝试输出二进制数据,请查看php的header函数,并确保设置了正确的content-type和content-disposition。查看该文件,关于使用/禁止输出缓冲区的更多信息。
尝试更改php.ini文件集“ output_buffering = Off”。您应该能够在使用系统命令而不是exec。页面上获得实时输出。系统命令将刷新输出
为什么不将输出通过管道传送到日志文件,然后使用该文件将内容返回给客户端。不太实时,但也许足够好?