Swoole4 provides a powerful CSP# for the PHP language ##Coroutine programming mode. The bottom layer provides 3 keywords, which can easily implement various functions. The PHP coroutine
-
Swoole4is borrowed fromGolang, and I would like to express my sincere thanks to theGOdevelopment team Tribute - PHP Swoole
Coroutines can complementGolangvery well.Golang: static language, rigorous, powerful and good in performance,PHP Swoole: dynamic language, flexible, simple and easy to use
Swoole-4.2. 9KeywordsandPHP-7.2.9version
- go
: Create a coroutine - chan
: Create a channel - defer
: Delay task, executed when the coroutine exits, first in, last out
3 functions are all memory operations, without any IO resource consumption. Just like PHPArray is very cheap. You can use it directly if necessary. This is different from socket and file operations. The latter need to apply for ports and file descriptors from the operating system, and reading and writing may cause blocking IO waiting.
PHP Video Tutorial"
go function to make a Functions are executed concurrently. During the programming process, if a certain piece of logic can be executed concurrently, it can be placed in the gocoroutine for execution.
function test1()
{
sleep(1);
echo "b";
}
function test2()
{
sleep(2);
echo "c";
}
test1();
test2();
Execution results: htf@LAPTOP-0K15EFQI:~$ time php b1.php
bc
real 0m3.080s
user 0m0.016s
sys 0m0.063s
htf@LAPTOP-0K15EFQI:~$
In the above code, test1 and test2 will be executed sequentially, which requires3It takes seconds to complete the execution.
go to create a coroutine can make the two functions test1 and test2 become concurrent executions .
Swoole\Runtime::enableCoroutine();
go(function ()
{
sleep(1);
echo "b";
});
go(function ()
{
sleep(2);
echo "c";
});
Swoole\Runtime::enableCoroutine()The function is to use thestream,sleep,# provided byPHP##pdo,mysqli,redisand other functions switch from synchronous blocking to asynchronous coroutineIOExecution results:
bchtf@LAPTOP-0K15EFQI:~$ time php co.php bc real 0m2.076s user 0m0.000s sys 0m0.078s htf@LAPTOP-0K15EFQI:~$
You can see that it only took
2 seconds to complete the execution.
- t1 t2 t3...
-
The concurrent execution time is equal to the execution time of all tasks The maximum value:max(t1, t2, t3, ...)
With
gokey After the word, concurrent programming is much simpler. At the same time, it brings new problems. If there are 2 coroutines executing concurrently, and another coroutine needs to rely on the execution results of these two coroutines, how to solve this problem? The answer is to use a channel (
). You can create a channel by using new chan in the Swoole4 coroutine. A channel can be understood as a queue with its own coroutine scheduling. It has two interfaces push and pop:
- push
- : writes content to the channel, if it is full, it It will enter the waiting state and automatically recover when there is space
- : Read the content from the channel. If it is empty, it will enter the waiting state and automatically recover when there is data
. $chan = new chan(2);
# 协程1
go (function () use ($chan) {
$result = [];
for ($i = 0; $i pop();
}
var_dump($result);
});
# 协程2
go(function () use ($chan) {
$cli = new Swoole\Coroutine\Http\Client('www.qq.com', 80);
$cli->set(['timeout' => 10]);
$cli->setHeaders([
'Host' => "www.qq.com",
"User-Agent" => 'Chrome/49.0.2587.3',
'Accept' => 'text/html,application/xhtml+xml,application/xml',
'Accept-Encoding' => 'gzip',
]);
$ret = $cli->get('/');
// $cli->body 响应内容过大,这里用 Http 状态码作为测试
$chan->push(['www.qq.com' => $cli->statusCode]);
});
# 协程3
go(function () use ($chan) {
$cli = new Swoole\Coroutine\Http\Client('www.163.com', 80);
$cli->set(['timeout' => 10]);
$cli->setHeaders([
'Host' => "www.163.com",
"User-Agent" => 'Chrome/49.0.2587.3',
'Accept' => 'text/html,application/xhtml+xml,application/xml',
'Accept-Encoding' => 'gzip',
]);
$ret = $cli->get('/');
// $cli->body 响应内容过大,这里用 Http 状态码作为测试
$chan->push(['www.163.com' => $cli->statusCode]);
});Execution results:
htf@LAPTOP-0K15EFQI:~/swoole-src/examples/5.0$ time php co2.php
array(2) {
["www.qq.com"]=>
int(302)
["www.163.com"]=>
int(200)
}
real 0m0.268s
user 0m0.016s
sys 0m0.109s
htf@LAPTOP-0K15EFQI:~/swoole-src/examples/5.0$
go is used here to create 3 coroutines, coroutines 2 and coroutines Cheng 3 requests the homepages of qq.com and 163.com respectively. Coroutine 1 needs to get the result of Http request. chan is used here to achieve concurrency management.
- 1
- Loop twice to
popthe channel, because the queue is empty, it will enter the waiting stateCoroutine2 - After the execution of the coroutine
3is completed, the data willpush. The coroutine1gets the result and continues to execute Delayed tasks
In coroutine programming, you may need to automatically execute some tasks and clean up when the coroutine exits. Similar to
register_shutdown_function of PHP, it can be implemented using defer in Swoole4. <pre class="brush:php;toolbar:false">Swoole\Runtime::enableCoroutine();
go(function () {
echo "a";
defer(function () {
echo "~a";
});
echo "b";
defer(function () {
echo "~b";
});
sleep(1);
echo "c";
});</pre>Execution results:
htf@LAPTOP-0K15EFQI:~/swoole-src/examples/5.0$ time php defer.php abc~b~a real 0m1.068s user 0m0.016s sys 0m0.047s htf@LAPTOP-0K15EFQI:~/swoole-src/examples/5.0$
Conclusion
Swoole4The Go Chan Defer provided isPHP It brings a new CSP concurrent programming model. Flexible use of various features provided by Swoole4 can solve the design and development of various complex functions at work.




























