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

Ste*_*ayi 393 php multithreading

有没有一种在PHP中实现多线程模型的现实方法,无论是真实的,还是仅仅模拟它.一段时间后,有人建议你可以强制操作系统加载另一个PHP可执行文件实例并处理其他同步进程.

这个问题是当PHP代码完成执行时,PHP实例仍然在内存中,因为没有办法从PHP中删除它.因此,如果您正在模拟几个线程,您可以想象会发生什么.所以我仍然在寻找一种可以在PHP中有效地完成或模拟多线程的方法.有任何想法吗?

Bab*_*aba 409

php中可以实现多线程

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

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

?>
Run Code Online (Sandbox Code Playgroud)

第一次运行

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
Run Code Online (Sandbox Code Playgroud)

第二轮

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
Run Code Online (Sandbox Code Playgroud)

真实世界的例子

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);
}
Run Code Online (Sandbox Code Playgroud)

  • 这很好,我多年没有接触PHP,现在它具有多线程功能! (13认同)
  • 请注意:pthreads 扩展的作者 Joe Watkins 停止了开发,转而支持新的并行扩展:https://github.com/krakjoe/pthreads/issues/929 (10认同)
  • 在这里下载windows binary http://windows.php.net/downloads/pecl/releases/pthreads/0.0.45/ (4认同)
  • @Baba,我无法在Xampp服务器上配置和安装pthread.你能帮帮我吗? (3认同)

小智 40

你为什么不用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]);
    }
}
Run Code Online (Sandbox Code Playgroud)

  • 比如@ e-info128说,这个实现分叉了进程,这意味着它运行在不同的进程上,并且不共享进程资源.话虽这么说,如果手边的工作不需要共享资源,那么这仍然可以工作,它将并行运行. (6认同)
  • 我正在使用上面的解决方案,并且工作正常,我认为这是使用php进行并行处理的最简单方法. (4认同)
  • 要将数据传递给它们,您也可以使用参数和 Redis 服务器。 (2认同)

小智 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"];
Run Code Online (Sandbox Code Playgroud)

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);
}
Run Code Online (Sandbox Code Playgroud)

verifysession.php

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

session_start();
var_dump($_SESSION);
Run Code Online (Sandbox Code Playgroud)

closeprocess.php

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

session_start();
$_SESSION["closesession"] = true;
var_dump($_SESSION);
Run Code Online (Sandbox Code Playgroud)

  • `set_time_limit(0);`yikes!永远不要这样做. (5认同)
  • 上次我检查(几年前)php不允许同时通过两个进程访问基于文件的会话存储.它锁定文件,第二个进程必须坐在那里等待第一个脚本停止.我说的是webserver环境,而不是CLI. (4认同)
  • @Kafoso 卡福索为什么不呢?好吧,我同意 PHP 作为 Web 脚本处理器,但为什么不在 CLI 中呢?如果出现问题,可以使用 Ctrl+C 终止 CLI... (2认同)
  • 我还会使用“set_time_limit(0);”来无限运行套接字侦听器。 (2认同)

J.D*_*ald 12

虽然你不能线程,但你确实在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做一个干净的退出.

  • @MaciejPaprocki pThread 现已从 php 7.4 中停止,而是使用并行 (3认同)

pin*_*sia 10

通过pthreads PECL扩展可以使用线程

http://www.php.net/manual/en/book.pthreads.php

  • “pThread 现已从 php 7.4 中停止,而是使用并行”。 (5认同)

Jas*_*vis 9

我知道这是一个老问题但是对于搜索的人来说,有一个用C语言编写的PECL扩展现在提供PHP多线程功能,它位于这里https://github.com/krakjoe/pthreads

  • pThread 现已从 php 7.4 中停止,而是使用并行 (4认同)

Ada*_*son 6

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

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

exec("/path/to/php -f '/path/to/file.php' | '/path/to/output.txt'");
Run Code Online (Sandbox Code Playgroud)

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


小智 6

您可以选择:

  1. 多卷曲
  2. 一个可以使用系统命令相同
  3. 理想的场景是,用 C 语言创建线程函数并用 PHP 编译/配置。现在该函数将是 PHP 的函数。


Jar*_*rod 6

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
    }

?>
Run Code Online (Sandbox Code Playgroud)


小智 5

你可以模拟线程.PHP可以通过popen(或proc_open)运行后台进程.这些过程可以通过stdin和stdout进行通信.当然这些进程本身可以是一个php程序.这可能就像你得到的那样接近.


She*_*ndu 5

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


Юри*_*нен 5

如果您使用的是 Linux 服务器,则可以使用

exec("nohup $php_path path/script.php > /dev/null 2>/dev/null &")
Run Code Online (Sandbox Code Playgroud)

如果你需要传递一些参数

exec("nohup $php_path path/script.php $args > /dev/null 2>/dev/null &")
Run Code Online (Sandbox Code Playgroud)

在脚本.php

$args = $argv[1];
Run Code Online (Sandbox Code Playgroud)

或者使用 Symfony https://symfony.com/doc/current/components/process.html

$process = Process::fromShellCommandline("php ".base_path('script.php'));
$process->setTimeout(0);     
$process->disableOutput();     
$process->start();
Run Code Online (Sandbox Code Playgroud)