发送邮件的时候,接口会卡住半天,需要异步执行代码,

即:接口数据直接返回,而邮件通知在后台运行

1、使用curl实现

$start_time = time();

$url = 'https://www.google.com/';

$ch = curl_init();
curl_setopt($ch, CURLOPT_URL, $url);
curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
curl_setopt($ch, CURLOPT_TIMEOUT, 1);
curl_exec($ch);
curl_close($ch);

$end_time = time();

echo 'done: ' . ($end_time - $start_time);

2、使用 Requests 实现

Requests 基于curl,所以和方式1是一样的

composer require rmccue/requests 
<?php

require './vendor/autoload.php';


$start_time = time();

$url = 'https://www.google.com/';

try {
$res = Requests::get($url, [], ['timeout' => 1]);
} catch (Exception $e) {
echo 'Exception' . PHP_EOL;
}

$end_time = time();
echo 'done: ' . ($end_time - $start_time);

以上两种方式都可以实现异步调用,缺点是必须有1秒的延时…

3、使用 register_shutdown_function 实现

<?php

class AsyncHook
{

private static $hook_list = array();
private static $hooked = false;

/**
* hook函数fastcgi_finish_request执行
*
* @param callback $callback
* @param array $params
*/
public static function hook($callback, $params = [])
{
self::$hook_list[] = array('callback' => $callback, 'params' => $params);

if (self::$hooked == false) {
self::$hooked = true;
register_shutdown_function(array(__CLASS__, '__run'));
}
}

/**
* 由系统调用
*
* @return void
*/
public static function __run()
{
if (function_exists('fastcgi_finish_request')) {
fastcgi_finish_request();
}

if (empty(self::$hook_list)) {
return;
}

foreach (self::$hook_list as $hook) {
$callback = $hook['callback'];
$params = $hook['params'];
call_user_func_array($callback, $params);
}
}
}

测试

function timer()
{
sleep(5);
echo 'timer done';
}


$start_time = time();

AsyncHook::hook('timer');

$end_time = time();

echo 'done: ' . ($end_time - $start_time) . PHP_EOL;
// done: 0
// timer done

该方法会继续执行代码,执行完成后并不会直接退出,而是会等待调用的方式都执行结束