1.8.0 使用内置Http异步客户端

最后更新于:2022-04-02 06:17:48

# 1.8.0 使用内置Http异步客户端 [TOC] Swoole-1.8.0版本内置了`HttpClient`,经过多个版本的迭代,内置HttpClient无论从功能、性能、稳定性上都已经非常出色。 ## 使用实例 ~~~ $cli = new swoole_http_client('127.0.0.1', 80); $cli->setHeaders(['User-Agent' => "swoole"]); $cli->post('/dump.php', array("test" => 'abc'), function ($cli) { echo $cli->body; }); ~~~ ## 并发能力 相比`curl`和`file_get_contents`这样PHP提供的Http客户端,`swoole_http_client`最大的优势是支持大量并发。 `file_get_contents`只能同时请求一个URL,并发只能通过开启多进程实现。`curl`提供了`curl_multi`功能实现并发基于`select`和多线程。并发能力都很差。而`swoole_http_client`是基于`epoll`实现的异步客户端,没有并发限制,可在一个进程内同时并发上万请求。 ## 性能问题 在PHP中也有纯PHP实现的Http客户端,如`Guzzle`,这些类库最大的问题是Http协议解析是由PHP代码实现的,PHP代码在这样场景下进行大量运算性能较差,而且还会占用大量内存。`swoole_http_client`是由C代码实现的,解析Http协议的性能是非常高的,内存占用也很少。 在解析`gzip`压缩后的HTML时,`swoole_http_client`的优势更为明显,它可以使用`download`方法,以很小的内存占用即可完成超大文件的下载。由于PHP层面没有提供`zlib`流式分段解压的支持,只能将Http Body全部放置到内存中,调用`gzdecode`一次性解压,而这会占用大量内存。 ## SSL支持 `swoole_http_client`支持SSL和TLS隧道加密的`https`网址,并且支持配置客户端证书。 ~~~ $cli = new swoole_http_client('127.0.0.1', 80, true); //如果服务器需要提供SSL证书 $cli->set(array( 'ssl_cert_file' => $certFile, 'ssl_key_file' => $keyFile, )); $cli->setHeaders(['User-Agent' => "swoole"]); $cli->get('/index.php', function ($cli) { file_put_contents(__DIR__.'/t.html', $cli->body); }); ~~~ ## Socks5代理 `swoole_http_client`支持`Socks5`代理,只需要设置几个参数就可以直接使用。 ~~~ $cli = new swoole_http_client('127.0.0.1', 80); $cli->set(array( 'socks5_host' => '192.168.1.100', 'socks5_port' => 1080, 'socks5_username' => 'username', //用户名和密码为可选项 'socks5_password' => 'password', )); $cli->setHeaders(['User-Agent' => "swoole"]); $cli->get('/index.php', function ($cli) { file_put_contents(__DIR__.'/t.html', $cli->body); }); ~~~ ## 上传文件 `swoole_http_client`底层使用了`sendfile`系统调用实现了http上传超大文件,配合底层的`epoll`可以实现非常低的消耗完成超巨大文件的上传。`sendfile`是零拷贝的,占用内存非常少,并且不存在多次内存复制开销。 ~~~ $cli = new swoole_http_client('127.0.0.1', 80); $cli->addFile(__DIR__.'/post.data', 'post'); $cli->addFile(dirname(__DIR__).'/test.jpg', 'debug'); $cli->post('/dump2.php', array("xxx" => 'abc', 'x2' => 'rango'), function ($cli) { echo $cli->body; $cli->close(); }); ~~~
';