有没有一种在PHP中实现多线程模型的现实方法,无论是真实的,还是仅仅模拟它.一段时间后,有人建议你可以强制操作系统加载另一个PHP可执行文件实例并处理其他同步进程.
这个问题是当PHP代码完成执行时,PHP实例仍然在内存中,因为没有办法从PHP中删除它.因此,如果您正在模拟几个线程,您可以想象会发生什么.所以我仍然在寻找一种可以在PHP中有效地完成或模拟多线程的方法.有任何想法吗?
是的,您可以使用pthreads在PHP中进行多线程处理
从PHP文档:
pthreads是一个面向对象的API,它提供了PHP中多线程所需的所有工具.PHP应用程序可以创建,读取,写入,执行和与Threads,Workers和Threaded对象同步.
警告:pthreads扩展名不能在Web服务器环境中使用.因此,PHP中的线程应仅保留给基于CLI的应用程序.
简单测试
#!/usr/bin/php arg = $arg; } public function run() { if ($this->arg) { $sleep = mt_rand(1, 10); printf('%s: %s -start -sleeps %d' . "\n", date("g:i:sa"), $this->arg, $sleep); sleep($sleep); printf('%s: %s -finish' . "\n", date("g:i:sa"), $this->arg); } } } // Create a array $stack = array(); //Initiate Multiple Thread foreach ( range("A", "D") as $i ) { $stack[] = new AsyncOperation($i); } // Start The Threads foreach ( $stack as $t ) { $t->start(); } ?>
第一次运行
12:00:06pm: A -start -sleeps 5 12:00:06pm: B -start -sleeps 3 12:00:06pm: C -start -sleeps 10 12:00:06pm: D -start -sleeps 2 12:00:08pm: D -finish 12:00:09pm: B -finish 12:00:11pm: A -finish 12:00:16pm: C -finish
第二轮
12:01:36pm: A -start -sleeps 6 12:01:36pm: B -start -sleeps 1 12:01:36pm: C -start -sleeps 2 12:01:36pm: D -start -sleeps 1 12:01:37pm: B -finish 12:01:37pm: D -finish 12:01:38pm: C -finish 12:01:42pm: A -finish
真实世界的例子
error_reporting(E_ALL); class AsyncWebRequest extends Thread { public $url; public $data; public function __construct($url) { $this->url = $url; } public function run() { if (($url = $this->url)) { /* * If a large amount of data is being requested, you might want to * fsockopen and read using usleep in between reads */ $this->data = file_get_contents($url); } else printf("Thread #%lu was not provided a URL\n", $this->getThreadId()); } } $t = microtime(true); $g = new AsyncWebRequest(sprintf("http://www.google.com/?q=%s", rand() * 10)); /* starting synchronization */ if ($g->start()) { printf("Request took %f seconds to start ", microtime(true) - $t); while ( $g->isRunning() ) { echo "."; usleep(100); } if ($g->join()) { printf(" and %f seconds to finish receiving %d bytes\n", microtime(true) - $t, strlen($g->data)); } else printf(" and %f seconds to finish, request failed\n", microtime(true) - $t); }
你为什么不用popen?
for ($i=0; $i<10; $i++) { // open ten processes for ($j=0; $j<10; $j++) { $pipe[$j] = popen('script2.php', 'w'); } // wait for them to finish for ($j=0; $j<10; ++$j) { pclose($pipe[$j]); } }
PHP中没有线程化,但是通过将HTTP请求用作异步调用,可以进行并发编程.
将curl的超时设置设置为1并为要彼此关联的进程使用相同的session_id,可以与会话变量进行通信,如下例所示.使用此方法,您甚至可以关闭浏览器,并且服务器上仍然存在并发进程.
不要忘记验证正确的会话ID,如下所示:
http://localhost/test/verifysession.php?sessionid = [正确的id]
$request = "http://localhost/test/process1.php?sessionid=".$_REQUEST["PHPSESSID"]; $ch = curl_init(); curl_setopt($ch, CURLOPT_URL, $request); curl_setopt($ch, CURLOPT_RETURNTRANSFER, true); curl_setopt($ch, CURLOPT_TIMEOUT, 1); curl_exec($ch); curl_close($ch); echo $_REQUEST["PHPSESSID"];
set_time_limit(0); if ($_REQUEST["sessionid"]) session_id($_REQUEST["sessionid"]); function checkclose() { global $_SESSION; if ($_SESSION["closesession"]) { unset($_SESSION["closesession"]); die(); } } while(!$close) { session_start(); $_SESSION["test"] = rand(); checkclose(); session_write_close(); sleep(5); }
if ($_REQUEST["sessionid"]) session_id($_REQUEST["sessionid"]); session_start(); var_dump($_SESSION);
if ($_REQUEST["sessionid"]) session_id($_REQUEST["sessionid"]); session_start(); $_SESSION["closesession"] = true; var_dump($_SESSION);
虽然你不能线程,但你确实在php中有一定程度的进程控制.这里有用的两个函数集是:
过程控制功能 http://www.php.net/manual/en/ref.pcntl.php
POSIX函数 http://www.php.net/manual/en/ref.posix.php
您可以使用pcntl_fork分叉您的进程 - 返回子进程的PID.然后你可以使用posix_kill来解析那个PID.
也就是说,如果你杀死父进程,应该向子进程发送一个信号告诉它死掉.如果php本身没有意识到这一点,你可以注册一个函数来管理它并使用pcntl_signal做一个干净的退出.
通过pthreads PECL扩展可以使用线程
http://www.php.net/manual/en/book.pthreads.php
我知道这是一个老问题但是对于搜索的人来说,有一个用C语言编写的PECL扩展现在提供PHP多线程功能,它位于这里https://github.com/krakjoe/pthreads
您可以使用exec()运行命令行脚本(例如命令行php),如果将输出通过管道传输到文件,则脚本将不会等待命令完成。
我不太记得php CLI语法,但是您想要这样的东西:
exec("/path/to/php -f '/path/to/file.php' | '/path/to/output.txt'");
我认为出于安全原因,很多共享托管服务器默认都禁用了exec(),但值得一试。
你可以模拟线程.PHP可以通过popen(或proc_open)运行后台进程.这些过程可以通过stdin和stdout进行通信.当然这些进程本身可以是一个php程序.这可能就像你得到的那样接近.