2

您可以使用 Guzzle 的Pool:batch()方法并行执行 http 请求。options它允许您使用第三个参数中的 key为请求设置默认选项。

但是,如果我需要针对池中的不同请求使用不同的选项怎么办?我想使用池执行 GET 请求并将每个响应流式传输到磁盘上的不同文件。有一个sink选项。但是如何将此选项的不同值应用于请求?

4

3 回答 3

10

Rastor 的示例几乎是正确的,但是如果您想为Pool()构造函数提供“选项”,则它的实现不正确。

他错过了这里提到的池选项数组的关键实现。

Guzzle 文档说:

当迭代器产生一个函数时,该函数将提供“request_options”数组,该数组应合并到任何现有选项之上,然后该函数必须返回一个可等待的承诺。

此外,如果您查看Pool()我链接到的评论下方的代码,您会看到 Guzzle 的 Pool 调用了可调用对象并将 Pool 的“选项”作为参数提供给它,这样您就应该将其应用于您的请求。

正确的优先级是

每个请求选项 > 池选项 > 客户端默认值。

如果您Pool()对象的选项数组应用于您的请求对象,您最终会遇到严重的错误,例如,如果您尝试将new Pool($client, $requests(100), ['options'=>['timeout'=>30.0]]);. 如果没有我更正的代码,则根本不会应用您的池选项,因为不支持正确合并池选项,因此最终放弃了它们。

所以是支持Pool()选项的正确代码:

<?php

$client = new \GuzzleHttp\Client();

$requests = function ($total) use ($client) {
    for ($i = 0; $i < $total; $i++) {
        $url = "domain.com/picture/{$i}.jpg";
        $filepath = "/tmp/{$i}.jpg";

        yield function($poolOpts) use ($client, $url, $filepath) {
            /** Apply options as follows:
             * Client() defaults are given the lowest priority
             * (they're used for any values you don't specify on
             * the request or the pool). The Pool() "options"
             * override the Client defaults. And the per-request
             * options ($reqOpts) override everything (both the
             * Pool and the Client defaults).
             * In short: Per-Request > Pool Defaults > Client Defaults.
             */
            $reqOpts = [
                'sink' => $filepath
            ];
            if (is_array($poolOpts) && count($poolOpts) > 0) {
                $reqOpts = array_merge($poolOpts, $reqOpts); // req > pool
            }

            return $client->getAsync($url, $reqOpts);
        };
    }
};

$pool = new Pool($client, $requests(100));

但是请注意,如果您知道永远不会向构造函数添加任何选项,则不必支持这些选项。在这种情况下,您可以查看官方 Guzzle 文档作为示例。Pool()new Pool()

官方示例如下:

// Using a closure that will return a promise once the pool calls the closure.
$client = new Client();

$requests = function ($total) use ($client) {
    $uri = '127.0.0.1:8126/guzzle-server/perf';
    for ($i = 0; $i < $total; $i++) {
        yield function() use ($client, $uri) {
            return $client->getAsync($uri);
        };
    }
};

$pool = new Pool($client, $requests(100));
于 2016-11-16T00:48:30.057 回答
0

对于大嘴 6

$client = new \GuzzleHttp\Client();

$requests = function ($total) use ($client) {
    for ($i = 0; $i < $total; $i++) {
        $url = "http://domain.com/picture/{$i}.jpg";
        $filepath = "/tmp/{$i}.jpg";

        yield function() use ($client, $url, $filepath) {
            return $client->getAsync($url, [
                'sink' => $filepath
            ]);
        };
    }
};

$pool = new Pool($client, $requests(100));
于 2016-11-07T14:33:39.463 回答
0

$options您可以在请求中单独指定您想要的。如果您将其传递给客户端,它将仅适用于所有请求。这是 Guzzle 6 文档的摘录:

创建客户端时,可以将标头添加为默认选项。当标头用作默认选项时,仅当正在创建的请求尚未包含特定标头时才应用它们。这包括在 send() 和 sendAsync() 方法中传递给客户端的请求以及客户端创建的请求(例如 request() 和 requestAsync())。

请参阅http://guzzle.readthedocs.org/en/latest/request-options.html?highlight=default#headers

于 2015-09-01T03:17:45.163 回答