如何在PHP应用程序中使用多线程


414

是否有一种实际的方法可以在PHP中实现多线程模型,无论是真正的还是仅对其进行仿真。一段时间以前,建议您可以强制操作系统加载PHP可执行文件的另一个实例并处理其他同时进行的进程。

问题在于,当PHP代码完成执行PHP实例时,它仍保留在内存中,因为无法从PHP中杀死它。因此,如果您正在模拟多个线程,则可以想象会发生什么。因此,我仍在寻找一种可以在PHP中有效完成或模拟多线程的方法。有任何想法吗?


1
见我的问题和答案在这里:stackoverflow.com/questions/2101640/...
powtac



也许感兴趣:pthreads.org
GibboK,2016年

现在是2020年,似乎我们想要的是“并行” php.net/manual/en/intro.parallel.php,而不是“ pthreads”:stackoverflow.com/a/56451969/470749
Ryan

Answers:


431

可以在PHP中进行多线程

是的,您可以使用pthreads在PHP中执行多线程

PHP文档

pthreads是一个面向对象的API,它提供了PHP中多线程所需的所有工具。PHP应用程序可以创建,读取,写入,执行和与线程,辅助对象和线程对象同步。

警告: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);
}

3
@Baba,我无法在Xampp服务器上配置和安装pthread。你能帮我吗?
Irfan 2013年

4
在此处下载Windows二进制文件windows.php.net/downloads/pecl/releases/pthreads/0.0.45
Baba

17
太好了,我多年没有接触PHP,现在它具有多线程功能!
cruizer 2014年

1
漂亮又简单!仅供参考,我正在Azure云Win服务器上部署一个应用程序,如果仅选择基本的1核心配置,则除非添加更多核心,否则多线程将不可用。
米兰

3
请注意:pthreads扩展的作者Joe Watkins中止了开发,转而使用了新的并行扩展:github.com/krakjoe/pthreads/issues/929
Anton Belonovich

42

你为什么不使用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]);
    }
}

4
我正在使用上述解决方案,并且工作正常,我认为这是使用php进行并行处理的最简单方法。
GodFather 2012年

7
就像@ e-info128所说的,此实现派生了该进程,这意味着它在不同的进程上运行,并且不共享进程资源。话虽如此,如果手头的工作不需要共享资源,那么它仍然可以工作,并且可以并行运行。
拉菲

1
不使用会话变量,如何传递变量来弹出?
atwellpub's

@atwellpub不可能,这些是不共享资源的独立进程。甚至会议也将是笨拙的IPC机制
Cholthi Paul Ttiopic '17

1
要将数据传递给他们,您也可以使用参数和Redis服务器。
阿米尔·佛

21

可用的PHP中没有线程,但是可以通过将HTTP请求用作异步调用来进行并发编程。

将curl的超时设置设置为1,并对要彼此关联的进程使用相同的session_id,则可以与会话变量进行通信,如下面的示例所示。使用这种方法,您甚至可以关闭浏览器,并发进程仍在服务器上。

不要忘记像这样验证正确的会话ID:

http://localhost/test/verifysession.php?sessionid = [正确的ID]

startprocess.php

$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"];

process1.php

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);
}

verifysession.php

if ($_REQUEST["sessionid"])
    session_id($_REQUEST["sessionid"]);

session_start();
var_dump($_SESSION);

closeprocess.php

if ($_REQUEST["sessionid"])
    session_id($_REQUEST["sessionid"]);

session_start();
$_SESSION["closesession"] = true;
var_dump($_SESSION);

4
上次我检查(几年前)时,php不允许两个进程同时访问基于文件的会话存储。它锁定文件,第二个进程必须坐在那里等待第一个脚本停止。我说的是网络服务器环境,而不是CLI。
Alexei Tenitski

5
set_time_limit(0);y!永远不要这样做。
卡夫索

@Kafoso Kafoso为什么不呢?我同意将PHP作为Web脚本处理器,但是为什么不在CLI中呢?如果出现问题,CLI可以使用Ctrl + C杀死...
sijanec

14

虽然您无法线程化,但您确实在php中具有一定程度的过程控制。这里有用的两个功能集是:

过程控制功能 http://www.php.net/manual/zh/ref.pcntl.php

POSIX功能 http://www.php.net/manual/zh/ref.posix.php

您可以使用pcntl_fork来分叉您的进程-返回子进程的PID。然后,您可以使用posix_kill丢弃该PID。

也就是说,如果您杀死了父进程,则应该向子进程发送一个信号,告诉它死亡。如果php本身无法识别,则可以注册一个函数来对其进行管理,并使用pcntl_signal进行干净退出。


1
这个答案现在已经过时了(知道它已经11岁了,这是很公平的)。查看下面的pthread。
Maciej Paprocki

@MaciejPaprocki pThread现在已从php 7.4中停用,请改为使用并行
Airy



5

您可以使用exec()运行命令行脚本(例如命令行php),如果将输出管道传输到文件,则脚本将不等待命令完成。

我不太记得php CLI语法,但是您想要这样的东西:

exec("/path/to/php -f '/path/to/file.php' | '/path/to/output.txt'");

我认为出于安全原因,很多共享托管服务器默认都禁用了exec(),但值得一试。


4

您可以模拟线程。PHP可以通过popen(或proc_open)运行后台进程。可以通过stdin和stdout与这些进程进行通信。当然,这些进程本身可以是php程序。那可能与您将获得的接近。


3

根据您要执行的操作,您还可以使用curl_multi来实现它。



3

您可以选择:

  1. 多卷
  2. 一个可以使用系统命令
  3. 理想的情况是用C语言创建一个线程函数并用PHP进行编译/配置。现在,该功能将成为PHP的功能。


3

pcntl_fork呢?

请查看我们的手册页以获取示例:PHP pcntl_fork

<?php

    $pid = pcntl_fork();
    if ($pid == -1) {
        die('could not fork');
    } else if ($pid) {
        // we are the parent
        pcntl_wait($status); //Protect against Zombie children
    } else {
        // we are the child
    }

?>

2

pcntl_fork如果安全模式已打开,则无法在Web服务器环境中使用。在这种情况下,它将仅在PHP的CLI版本中起作用。


1

如果您使用的是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();

-1

在撰写当前评论时,我对PHP线程一无所知。我来这里亲自寻找答案,但是一种解决方法是,从Web服务器接收请求的PHP程序将整个答案公式委托给控制台应用程序,该控制台应用程序将其输出,请求的答案存储到二进制文件中。并且启动控制台应用程序的PHP程序逐字节返回该二进制文件,作为对收到的请求的答复。控制台应用程序可以用服务器上运行的任何编程语言编写,包括那些具有适当线程支持的语言,包括使用OpenMP的C ++程序。

一种不可靠,肮脏的技巧是使用PHP执行控制台应用程序“ uname”,

uname -a

并将该控制台命令的输出打印到HTML输出中,以查找服务器软件的确切版本。然后,将完全相同版本的软件安装到VirtualBox实例,编译/组装所需的任何完全独立的(最好是静态的)二进制文件,然后将其上传到服务器。从那时起,PHP应用程序可以使用具有适当多线程功能的控制台应用程序中的那些二进制文件。当服务器管理员未将所有必需的编程语言实现安装到服务器时,这是一种肮脏,不可靠的解决方法。需要注意的是,PHP应用程序收到控制台应用程序的每个请求都会终止/退出/ get_killed。

至于托管服务管理员对这种服务器使用模式的看法,我想这可以归结为文化。在北欧,服务提供商必须提供广告,并且如果允许执行控制台命令并且允许上传非恶意软件文件,并且服务提供商有权在几分钟甚至30秒后终止任何服务器进程,那么托管服务管理员就没有理由提出适当的投诉。在美国和西欧,情况/文化大不相同,我认为在美国和/或西欧,托管服务提供商很有可能会拒绝为使用上述技巧的托管服务客户提供服务。鉴于我在美国的亲身经历,这只是我的猜测 托管服务,以及我从其他人那里听说的有关西欧托管服务的信息。截至我撰写当前评论(2018_09_01)时,我对南欧托管服务提供商,南欧网络管理员的文化规范一无所知。


-3

可能是我错过了一些东西,但是exec在我在Windows中使用的Windows环境中对我来说不是异步的,它像魅力一样工作;)

$script_exec = "c:/php/php.exe c:/path/my_ascyn_script.php";

pclose(popen("start /B ". $script_exec, "r"));

1
要破解PHP中的多线程,您应该打开多个popen()语句(PHP不会等待它完成)。然后在打开大约六点后,在其中调用pclose()(PHP将等待pclose()完成)。在您的代码中,您在打开每个线程后立即将其关闭,因此您的代码不会表现为多线程。
Kmeixner 2014年

-5

多线程意味着同时执行多个任务或进程,虽然没有直接的方法可以在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中实现多线程。

这个家伙在php中做得非常好


此外,这与MultiThreading无关。这是并行处理。完全不同的事情。
Digital Human

我认为这是一种应急方案,提供的解决方案背后的想法非常合适,但是我猜想不同的人可能对“真正的多线程”的构成产生激烈的争论,因为并发和基于硬件的区别并行处理,如:youtube.com/watch?
v=cN_DpYBzKso
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.