是否有一种在PHP中实现多线程模型的现实方法,无论是真实的还是仅仅模拟它。一段时间后,有人建议您可以强制操作系统加载另一个PHP可执行文件实例并处理其他同时进程。
这个问题是当PHP代码完成执行时,PHP实例仍然在内存中,因为没有办法从PHP中删除它。因此,如果您正在模拟几个线程,您可以想象会发生什么。所以我仍然在寻找一种可以在PHP中有效地完成或模拟多线程的方法。有什么想法吗?
答案 0 :(得分:389)
是的,您可以使用pthreads
在PHP中进行多线程处理pthreads是一个面向对象的API,它提供了PHP中多线程所需的所有工具。 PHP应用程序可以创建,读取,写入,执行和与Threads,Workers和Threaded对象同步。
警告强>: pthreads扩展不能在Web服务器环境中使用。因此,PHP中的线程应保留在基于CLI的应用程序中。
简单测试
#!/usr/bin/php
<?php
class AsyncOperation extends Thread {
public function __construct($arg) {
$this->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);
}
答案 1 :(得分:34)
为什么不使用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]);
}
}
答案 2 :(得分:17)
PHP中没有线程化,但是通过将HTTP请求用作异步调用,可以进行并发编程。
将curl的超时设置设置为1,并为要彼此关联的进程使用相同的session_id,您可以与会话变量进行通信,如下例所示。使用此方法,您甚至可以关闭浏览器,并且服务器上仍然存在并发进程。
不要忘记验证正确的会话ID,如下所示:
http://localhost/test/verifysession.php?sessionid=[the正确的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);
答案 3 :(得分:11)
虽然你不能线程,但你确实在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做一个干净的退出。
答案 4 :(得分:8)
答案 5 :(得分:8)
我知道这是一个老问题但是对于搜索的人来说,有一个用C语言编写的PECL扩展现在提供PHP多线程功能,它位于https://github.com/krakjoe/pthreads
答案 6 :(得分:5)
您可以使用exec()运行命令行脚本(例如命令行php),如果将输出传递给文件,那么您的脚本将不会等待命令完成。
我不太清楚php CLI的语法,但你需要这样的东西:
exec("/path/to/php -f '/path/to/file.php' | '/path/to/output.txt'");
我认为出于安全原因,默认情况下会禁用多个共享托管服务器exec(),但可能值得一试。
答案 7 :(得分:5)
您可以模拟线程。 PHP可以通过popen(或proc_open)运行后台进程。这些过程可以通过stdin和stdout进行通信。当然这些进程本身可以是一个php程序。这可能就像你得到的那样接近。
答案 8 :(得分:3)
如果您使用的是Linux服务器,则可以使用
exec("nohup $php_path path/script.php > /dev/null 2>/dev/null &")
如果您需要传递一些参数
exec("nohup $php_path path/script.php $args > /dev/null 2>/dev/null &")
在script.php中
$args = $argv[1];
或使用Symfony https://symfony.com/doc/current/components/process.html
$process = Process::fromShellCommandline("php ".base_path('script.php'));
$process->setTimeout(0);
$process->disableOutput();
$process->start();
答案 9 :(得分:3)
pcntl_fork怎么样?
查看我们的手册页以获取示例:PHP pcntl_fork
答案 10 :(得分:3)
我知道这已经过时了,但您可以查看http://phpthreadlib.sourceforge.net/
它支持双向线程间通信,并具有内置保护功能,可以杀死子线程(防止孤儿)。
答案 11 :(得分:3)
根据您的尝试,您还可以使用curl_multi来实现它。
答案 12 :(得分:2)
pcntl_fork
已启用安全模式,则
{{1}}将无法在Web服务器环境中运行。在这种情况下,它只能在PHP的CLI版本中使用。
答案 13 :(得分:2)
您可以选择:
答案 14 :(得分:2)
由于PECLpthreads≥2.0.0,Thread class可用。
答案 15 :(得分:0)
在撰写当前评论时,我不了解PHP线程。我来这里亲自寻找答案,但是一种解决方法是,从Web服务器接收请求的PHP程序将整个答案公式委托给控制台应用程序,该控制台应用程序将其输出,请求的答案存储到二进制文件中。并且启动控制台应用程序的PHP程序逐字节返回该二进制文件,作为对收到的请求的答复。控制台应用程序可以用服务器上运行的任何编程语言编写,包括那些具有适当线程支持的语言,包括使用OpenMP的C ++程序。
一个不可靠,肮脏的技巧是使用PHP执行控制台应用程序“ uname”
uname -a
,然后将该控制台命令的输出打印到HTML输出中,以查找服务器软件的确切版本。然后,将完全相同版本的软件安装到VirtualBox实例,编译/组装所需的任何完全独立的(最好是静态的)二进制文件,然后将其上传到服务器。从那时起,PHP应用程序可以在具有适当多线程功能的控制台应用程序中使用那些二进制文件。当服务器管理员未将所有必需的编程语言实现安装到服务器时,这是一种肮脏,不可靠的解决方法。需要注意的是,PHP应用程序收到控制台应用程序的每个请求都会终止/退出/ get_killed。
关于托管服务管理员对这种服务器使用模式的看法,我认为这可以归结为文化。在北欧,服务提供商必须提供广告,并且如果允许执行控制台命令并且允许上传非恶意软件文件,并且服务提供商有权在几分钟甚至30秒后终止任何服务器进程,那么托管服务管理员就没有理由提出适当的投诉。在美国和西欧,情况/文化大不相同,我相信在美国和/或西欧,托管服务提供商很有可能会 拒绝为使用上述技巧的托管服务客户提供服务。考虑到我在美国托管服务方面的个人经验以及我从其他人那里听说的有关西欧托管服务的信息,这只是我的猜测。截至我撰写当前评论(2018_09_01)时,我对南欧托管服务提供商,南欧网络管理员的文化规范一无所知。
答案 16 :(得分:-2)
可能是我错过了一些东西但是exec在windows环境中并没有像我一样异步我在Windows中使用以下它就像魅力一样;)
$script_exec = "c:/php/php.exe c:/path/my_ascyn_script.php";
pclose(popen("start /B ". $script_exec, "r"));
答案 17 :(得分:-4)
多线程意味着同时执行多个任务或进程,我们可以通过使用以下代码在php中实现这一点,虽然没有直接的方法在php中实现多线程,但我们可以通过以下方式实现几乎相同的结果。
chdir(dirname(__FILE__)); //if you want to run this file as cron job
for ($i = 0; $i < 2; $i += 1){
exec("php test_1.php $i > test.txt &");
//this will execute test_1.php and will leave this process executing in the background and will go
//to next iteration of the loop immediately without waiting the completion of the script in the
//test_1.php , $i is passed as argument .
}
Test_1.php
$conn=mysql_connect($host,$user,$pass);
$db=mysql_select_db($db);
$i = $argv[1]; //this is the argument passed from index.php file
for($j = 0;$j<5000; $j ++)
{
mysql_query("insert into test set
id='$i',
comment='test',
datetime=NOW() ");
}
这将同时执行test_1.php两次,两个进程将同时在后台运行,这样就可以在php中实现多线程。
这家伙做得非常好Multithreading in php