线程在php5中不能一起工作

时间:2013-10-25 15:40:01

标签: php multithreading

我正在尝试用php实现multi_threading程序,我已经下载了pthread包并完成了所有必需的配置。 然后我试着运行这段代码:

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

//Iniciate Miltiple Thread
foreach ( range("A", "D") as $i ) {
    $stack[] = new AsyncOperation($i);
}

// Start The Threads
foreach ( $stack as $t ) {
    $t->start();
}

?>

但每次我运行代码时都会得到类似的东西:

4:29:47pm: A -start -sleeps 1 
4:29:48pm: A -finish 
4:29:47pm: C -start -sleeps 3 
4:29:50pm: C -finish 
4:29:47pm: D -start -sleeps 4 
4:29:51pm: D -finish 
4:29:47pm: B -start -sleeps 7 
4:29:54pm: B -finish

另一次运行:

5:29:21pm: A -start -sleeps 1 
5:29:22pm: A -finish 
5:29:21pm: B -start -sleeps 3 
5:29:24pm: B -finish 
5:29:21pm: D -start -sleeps 8 
5:29:29pm: D -finish 
5:29:21pm: C -start -sleeps 9 5:29:30pm: C -finish

正如您所看到的那样,线程似乎无法同时工作的问题,无论何时线程启动,其他线程在完成之前都没有启动。我注意到的有趣的事情是线程总是从最小的睡眠时间开始排序!

我希望有些人会这样:

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

有人可以告诉我如何获得想要的运行?我正在使用php5。 注意:我从本网站的其他帖子获得了代码: check it up

提前多多感谢。

2 个答案:

答案 0 :(得分:1)

嗯......看着时间戳,看起来它们都是在同一时间开始的(下午4:29:47):

4:29:47pm: A -start -sleeps 1 
4:29:48pm: A -finish 
4:29:47pm: C -start -sleeps 3 
4:29:50pm: C -finish 
4:29:47pm: D -start -sleeps 4 
4:29:51pm: D -finish 
4:29:47pm: B -start -sleeps 7 
4:29:54pm: B -finish

与此相同(下午5:29:21):

5:29:21pm: A -start -sleeps 1 
5:29:22pm: A -finish 
5:29:21pm: B -start -sleeps 3 
5:29:24pm: B -finish 
5:29:21pm: D -start -sleeps 8 
5:29:29pm: D -finish 
5:29:21pm: C -start -sleeps 9
5:29:30pm: C -finish

我错过了什么吗?

我确实看到他们在睡觉前没有打印出来。

答案 1 :(得分:1)

正如马史密斯所提到的,时间戳表明执行确实同时开始。当您打印到多个线程的输出时,您很幸运能够读取它,从来没有理解数据的顺序......如果数据的顺序非常重要,您应该使用互斥锁进行打印。

另一方面,sleep()不适合在多线程应用程序中使用 - 理论上,睡眠是针对进程的,并且应该导致整个进程休眠。我理论上说因为posix是同一个想法的百万个实现,然后有窗口要考虑,我没有足够的经验与windows提供合理的建议,但你不应该依赖它的行为......

usleep函数更适合线程:

<?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);
        usleep($sleep*1000000);
        printf('%s: %s  -finish' . "\n", date("g:i:sa"), $this->arg);
    }
}
}

// Create a array
$stack = array();

//Iniciate Miltiple Thread
foreach ( range("A", "D") as $i ) {
    $stack[] = new AsyncOperation($i);
}

// Start The Threads
foreach ( $stack as $t ) {
    $t->start();
}

foreach ($stack as $t)
        $t->join();
?>

另外,你应该加入加入线程的习惯而不允许php为你清理它,这是一个很好的习惯,因为它可以保持一切与你期望的一致。

即使是睡眠,对于使用多线程的真实生产代码来说还不够好,问题在于它不会使线程处于接受状态:如果你指示一个线程睡眠,其他线程都不会影响它半小时,你无能为力(优雅)唤醒它。 pthreads带有高级同步,你应该在任何实际代码中使用它。

<强>参考文献: