Note: Part 2 of this post can be found here.
Of all the glorious programming languages in existence, you’ve chosen to work with PHP. Okay, maybe you were forced to, but that doesn’t mean you can’t have fun right? Hmm, fun… What’s fun? Threading, race conditions, and deadlocks. Sounds like loads of fun! But alas, PHP doesn’t have functionality to cover this. True, but the host operating system does. We can just fork processes manually like in the good ‘ole days and then ride off into the sunset. But we shouldn’t do that, it’s wrong. It’s taking advantage of PHP. Bah! We’re going to do it anyways. And FOR THE LOVE OF GOD, do not do this in production code. If you need multi-threading, use a different language.
Process Control
One of those things that you are bound to learn about if you go through a computer science degree is process control. Process control usually comes up slowly and then bites you in the ass REALLY hard during operating systems courses. Thinking about race conditions, deadlocks, and collisions are all part of the game. And guess what? When you are writing multi-threaded programs this is exactly the type of thing you will encounter.
When thinking about process control, people often cite some form of a producer-consumer problem. You have one process (thread) producing information, and another thread consuming it. For instance, a producer may spit out a stream of integers, and the consumer will, well, consume them. However, the consumer will terminate itself when it hits a non-integer character. That’s called out exit condition, and it’s very important. You have to always remember to have some sort of fool-proof exit condition, otherwise the process is going to hang and go zombie on your. If your program get’s executed a few thousand times a day, you get a few thousand zombies (process zombie apocalypse!). So, the import part here is to remember that you need to have exit condtions.
Getting Started
The first step into multi-threading in PHP is making sure your build is compiled with the –enable-pcntl flag set. If that’s not set, you’re dead in the water. Otherwise, we can take a look at a simple example.
1
2
3
4
5
6
| $processID = pcntl_fork();
if($processID) {
echo "I'm in the parent process!";
} else {
echo "I'm in the child process!";
} |
$processID = pcntl_fork();
if($processID) {
echo "I'm in the parent process!";
} else {
echo "I'm in the child process!";
}
Now, a quick explanation. The pcntl_fork() function takes the current running process and makes a copy of it. Everything (for practical purposes) is copied into this new child process except the process id(pid) which is changed to something new. This is where things get a bit weird. When pcntl_fork() executes, the pid of the child process is returned to the parents thread of execution. A value of 0(zero) is returned to the child process’ thread of execution. Since you can differentiate between threads of execution, you can make them do different things with a simple if statement like above. So what does this program print?
I'm in the parent process!
I'm in the child process!
or
I'm in the child process!
I'm in the parent process!
It can actually go either way here. It depends entirely on how your operating system decided to schedule the processes. But remember, the child process is an exact (not entirely true, but let’s go with it) copy of the parent process. So what happens if we change the code a bit.
1
2
3
4
5
6
7
| $processID = pcntl_fork();
if($processID) {
echo "I'm in the parent process!";
} else {
echo "I'm in the child process!";
}
echo "End of the line folks."; |
$processID = pcntl_fork();
if($processID) {
echo "I'm in the parent process!";
} else {
echo "I'm in the child process!";
}
echo "End of the line folks.";
Since the last echo statement exists in both parent and child processes, you’ll get:
I'm in the parent process!
End of the line folks.
I'm in the child process!
End of the line folks.
Waiting for the Child
One of the problems that you run in to with writing multi-threaded programs is that the child process can finish before the parent, or the parent can finish before the child. You just never really know. So you need a way to wait for processes to finish. Lucky for us, PHP at least provides this functionality for us.
1
2
3
4
5
6
7
| $pid=pcntl_fork();
if($pid) {
pctnl_waitpid($pid,$status,WUNTRACED);
echo "In parent process!";
} else {
echo "In child process!";
} |
$pid=pcntl_fork();
if($pid) {
pctnl_waitpid($pid,$status,WUNTRACED);
echo "In parent process!";
} else {
echo "In child process!";
}
By using the pcntl_waitpid() function, we can force the parent process to wait for the child process to finish executing. This is handy for if you have a critical procedure that your child process must complete before the parent can continue. In our case, the output will always be:
In child process!
In parent process!
Next Time…
That’s all for now, but soon I’ll have The 2nd (and last) part of this article up for your enjoyment. In that part, we’ll cover some more advanced notions like getting/setting priority, setting alarms, and signal processing. You should follow me on Twitter and/or become a fan on Facebook to find out when this exciting(?) article comes out.