gpt4 book ai didi

php - 从 PHP 执行程序挂起 APACHE

转载 作者:可可西里 更新时间:2023-11-01 11:57:50 24 4
gpt4 key购买 nike

您好,提前感谢您的关注。

在过去的两周里,我一直在与一些让我发疯的事情作斗争。我在我的 Windows 机器上安装了 APACHE (2.2.22) 和 PHP (5.4.3),我试图从同时调用另一个程序的 PHP 脚本调用一个程序。这两个程序都是用 C/C++ 编写的,并使用 MINGW32 编译的。关于 Windows 版本,我测试过 Windows 2003 Server 和 Windows 7 Professional,两者都给我同样的问题。

让我介绍一下这两个程序:

1) mytask.exe:这是一个在后台执行并定期将其状态填充到文件中的程序。

2) job.exe:这是我要从 PHP 脚本调用的程序。它的目标是将 mytask.exe 作为独立进程(而不是线程)生成。

如果我从控制台窗口运行下面的命令,那么 job.exe 会立即返回并让 mytask.exe 在后台运行直到它终止。

> job.exe spawn mytask.exe
jobid=18874111458879FED

请注意,job.exe 转储用于管理 mytask.exe 的标识符。例如:

> job.exe status 18874111458879FED
RUNNING

我已经检查过,如果我从 PHP 脚本运行第一个命令,PHP 脚本将随机永远阻塞。如果我查看 Windows 的任务管理器,我可以看到 job.exe 处于类似僵尸的状态。我可以断言 job.exe 在其 main() 例程中有效地达到了通常的 return 0; 语句,所以它似乎是在 C 运行时中的某些东西.此外,如果我编写一个简单的 mytask.exe,它只休眠 10 秒,那么 PHP 脚本也会阻塞 10 秒(或者在我刚才提到的随机行为之后永远阻塞)。换句话说,当我从 PHP 脚本调用 job.exe 时,我无法让 job.exe 在不等待它结束的情况下生成一个进程。

所以:在生成 mytask.exe 时我做错了什么,现在,这是题外话的第二部分。

我使用 WINAPI 函数 CreateProcess()从 job.exe 生成任务。根据 MSDN 文档,我使用 bInheritHandles = FALSE 调用 CreateProcess,以避免子进程使用 PHP 脚本产生 I/O 死锁。我还关闭了 PROCESS_INFORMATION 结构中 CreateProcess() 返回的进程句柄。我唯一不做的就是等待进程结束。另一方面,关于 PHP 方面,我尝试了 exec()proc_open() PHP 函数来调用 job.exe 但没有成功。

不过,我最后的观察结果似乎是正确的,但它们并没有说服我,因为我不明白为什么它们会以某种方式起作用。事实上,如果 mytask.exe 在休眠前执行 fclose(stdout),则 PHP 脚本会立即返回。但是,如何???我告诉 CreateProcess() 不要继承句柄,那么为什么我会得到这些结果?无论如何,我不能坚持使用这个补丁,因为由 job.exe 启动的程序可能不知道谁在调用它们,所以关闭这些程序的标准输出不是一个好的解决方案。在 UNIX 中,事情是如此简单……只需调用 fork(),关闭标准流,然后调用 execve 来调用程序。在 Windows 中,我还尝试使用 CreateThread() 创建一个包装器线程(模拟 fork()),然后在关闭标准流后从该线程调用 CreateProcess() ......但这也关闭了 job.exe 的流!

所有这些问题都可以综合为一个问题:如何从 PHP 执行创建其他进程的程序?

我希望有人能阐明这个问题...非常感谢!

最佳答案

我想我已经确定了解决方案,它分为两部分:

1) 关于主进程停止直到子进程结束这一事实。

根据 MSDN 文档,这是 CreateProcess() 的定义:

BOOL WINAPI CreateProcess(
_In_opt_ LPCTSTR lpApplicationName,
_Inout_opt_ LPTSTR lpCommandLine,
_In_opt_ LPSECURITY_ATTRIBUTES lpProcessAttributes,
_In_opt_ LPSECURITY_ATTRIBUTES lpThreadAttributes,
_In_ BOOL bInheritHandles,
_In_ DWORD dwCreationFlags,
_In_opt_ LPVOID lpEnvironment,
_In_opt_ LPCTSTR lpCurrentDirectory,
_In_ LPSTARTUPINFO lpStartupInfo,
_Out_ LPPROCESS_INFORMATION lpProcessInformation
);

正如我在问题中所说,我将 FALSE 传递给 bInheritHandles,但我也将 0 传递给 dwCreationFlags。经过更多研究后,我发现有一个名为 DETACHED_PROCESS 的标志,MSDN 对此表示:

For console processes, the new process does not inherit its parent's console (the default). The new process can call the AllocConsole function at a later time to create a console. For more information, see Creation of a Console.

现在,job.exe 立即返回,尽 pipe 进程继续执行。

2) 关于PHP脚本在调用exec()时随机挂起的事实

这似乎是 PHP 的一个错误。在 PHP session 上下文中运行 exec() 系列函数可能会使 APACHE 随机挂起,需要重新启动服务器。我找到了一个 thread in the Internet其中用户注意到在调用 exec() 之前关闭 session (通过 session_write_close())将防止脚本挂起。这同样适用于 proc_open/proc_close 函数。所以,我的脚本现在看起来像这样:

session_write_close();  //Close the session before proc_open()
$proc = proc_open($cmd,$pipedesc,$pipes);
//do stuff with pipes...
//... and close pipes
$retval = proc_close($proc);
session_start(); //restore session

希望这对您有所帮助。

关于php - 从 PHP 执行程序挂起 APACHE,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/13690490/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com