PHP CLI进程在退出时永远挂起

时间:2014-11-12 10:11:07

标签: php zeromq command-line-interface strace

我有一个PHP脚本,我在命令行上运行,当脚本应该退出时,它只会永远挂起。它发生在Windows和Linux上,因此它与操作系统无关。

我尝试使用XDebug调试代码,进入最后的exit语句(已尝试die但没有运气)。在对某些对象运行了几个析构函数之后,没有什么可以逐步完成的,并且该进程只是永远等待。它没有做任何事情,没有消耗任何资源。让进程退出的唯一方法是杀死它,例如Ctrl+C

在我的所有脚本中都没有发生,但我有一个案例可以重现。我不确定它是否在我的代码,库代码(主要是Symfony,Doctrince等)或PHP本身。

我使用strace运行脚本,输出的结尾附在下面。我不知道如何调试此输出,但似乎PHP正在查询最后一个语句中的内容。

如何进一步调试?任何帮助将不胜感激。

strace输出:

fstat(12, {st_mode=S_IFREG|0777, st_size=1314, ...}) = 0
fstat(12, {st_mode=S_IFREG|0777, st_size=1314, ...}) = 0
fstat(12, {st_mode=S_IFREG|0777, st_size=1314, ...}) = 0
fstat(12, {st_mode=S_IFREG|0777, st_size=1314, ...}) = 0
mmap(NULL, 1314, PROT_READ, MAP_SHARED, 12, 0) = 0x7fb8b969c000
munmap(0x7fb8b969c000, 1314)            = 0
close(12)                               = 0
umask(022)                              = 022
close(3)                                = 0
close(4)                                = 0
write(11, "\1\0\0\0\1", 5)              = 5
shutdown(11, SHUT_RDWR)                 = 0
close(11)                               = 0
write(6, "\1\0\0\0\0\0\0\0", 8)         = 8
close(2)                                = 0
close(1)                                = 0
munmap(0x7fb8b969a000, 4096)            = 0
close(0)                                = 0
munmap(0x7fb8b969b000, 4096)            = 0
munmap(0x7fb8b3191000, 790528)          = 0
munmap(0x7fb8b32d3000, 266240)          = 0
munmap(0x7fb8b3314000, 266240)          = 0
munmap(0x7fb8b3355000, 266240)          = 0
munmap(0x7fb8b3396000, 266240)          = 0
munmap(0x7fb8b33d7000, 266240)          = 0
munmap(0x7fb8b3418000, 266240)          = 0
munmap(0x7fb8b3459000, 266240)          = 0
munmap(0x7fb8b349a000, 266240)          = 0
munmap(0x7fb8b34db000, 266240)          = 0
munmap(0x7fb8b351c000, 266240)          = 0
munmap(0x7fb8b94ba000, 266240)          = 0
write(10, "\1\0\0\0\0\0\0\0", 8)        = 8
poll([{fd=5, events=POLLIN}], 1, 4294967295Process 1358 detached
 <detached ...>

1 个答案:

答案 0 :(得分:6)

我发现该问题与ZMQ扩展程序有关。我的脚本试图向不存在的主机发送消息,即使在我的脚本末尾断开套接字时,套接字也在等待尝试发送数据。

对于任何遇到相同问题的人,可以通过将套接字上的\ZMQ::SOCKOPT_LINGER选项设置为较低的值来解决,例如

<?php

$context = new \ZMQContext(1);
$socket = new \ZMQSocket($context, \ZMQ::SOCKET_PUSH);
$dsn = 'tcp://127.0.0.1:1337';
$socket->connect($dsn);
$socket->send('hi');
echo 'Message sent' . PHP_EOL;

// Without this line, the script will wait forever after the exit statement
$socket->setSockOpt(\ZMQ::SOCKOPT_LINGER, 1000);
$socket->disconnect($dsn);
echo 'Socket disconnected' . PHP_EOL;
exit();