0

0

PHP用Guzzle怎样设重试调用服务_PHPGuzzle设重试调用法【稳健】

看不見的法師

看不見的法師

发布时间:2026-02-08 14:21:09

|

649人浏览过

|

来源于php中文网

原创

最直接方式是使用 GuzzleHttp\Middleware::retry() 创建重试中间件并注入 HandlerStack;需自定义 $decider 判定网络异常和5xx/429响应才重试,$delay 实现指数退避,且必须通过 HandlerStack 传入 Client 而非直接赋值 handler。

php用guzzle怎样设重试调用服务_phpguzzle设重试调用法【稳健】

GuzzleHttp\Client 配置重试中间件最直接的方式

PHP 里 Guzzle 本身不内置重试逻辑,必须靠 RetryMiddleware + HandlerStack 手动组装。官方推荐做法是用 GuzzleHttp\Middleware::retry(),它返回一个可插入的中间件函数。

关键点:不能只改 timeoutconnect_timeout,那只是单次请求超时控制,和“失败后重试”是两回事。

  • retry 中间件需绑定到 HandlerStack,再传给 Client 构造函数
  • 重试判定逻辑由回调函数 $decider 控制,不是所有 HTTP 状态码都该重试(比如 400、401、403 通常不该重试)
  • 指数退避由 $delay 回调决定,Guzzle 不自动帮你算 2^n * 100ms 这类值,得自己写

$decider 怎么写才不乱重试

默认的 retry 决策函数对网络错误(如连接拒绝、超时)返回 true,但对 5xx 响应默认不重试——这点常被忽略,导致服务端临时 502/503 时没重试就直接报错。

典型安全策略:重试网络异常 + 5xx 响应,跳过 4xx(除 429):

立即学习PHP免费学习笔记(深入)”;

Eclipse中创建Web Maven Project并部署到Tomcat中 中文WORD版
Eclipse中创建Web Maven Project并部署到Tomcat中 中文WORD版

大家都知道,在进行J2EE项目的开发过程中,在调试阶段如果只是修改了页面是不需要重启应用服务器的,比如不需要重启Tomcat。只需要在浏览器中 进行页面刷新即可。其实之所以不用重启Tomcat等应用服务器,其根本原因是因为我们可以在应用服务器的配置文件中设置虚拟目录,这样就可以知道web 项目所在的目录,于是就可以省去打包、然后再重新发布到服务器的步骤。感兴趣的朋友可以过来看看

下载
$decider = function ($retries, $request, $response, $exception) {
    // 有异常:连接失败、超时、DNS 错误等
    if ($exception instanceof \Exception) {
        return $retries < 3;
    }
    // 有响应但状态码是 5xx 或 429
    if ($response && $response->getStatusCode() >= 500 || $response->getStatusCode() === 429) {
        return $retries < 3;
    }
    return false;
};
  • 别把 400401404 加进重试,它们代表客户端问题,重试无意义
  • 注意 $exception 可能为 null$response 可能为 null,判空前先检查
  • 重试次数建议设为 2~3 次,太多会放大下游压力

怎么让重试带延迟且不阻塞主线程

Guzzle 是同步 HTTP 客户端,所谓“不阻塞”是指避免固定 sleep,而用指数退避降低重试冲击。Guzzle 的 $delay 参数接收一个回调,返回毫秒数:

$delay = function ($retries) {
    return (int) pow(2, $retries) * 100; // 第1次100ms,第2次200ms,第3次400ms
};
  • 不要用 sleep(1) 这种硬延迟,它会让整个 PHP 进程挂起,影响并发能力
  • 实际延迟是累加在每次重试前的,Guzzle 会自动调用 usleep(),你只需返回整数毫秒
  • 如果用了 curl handler(默认),延迟期间 CPU 是空闲的,不影响其他请求处理

guzzlehttp/guzzle 7.x 时要注意 HandlerStack 初始化方式

7.x 版本中 HandlerStack::create() 返回的默认已含重定向、cookie、http_errors 等中间件,你只需 prepend retry,别 replace 整个栈:

$stack = HandlerStack::create();
$stack->push(Middleware::retry($decider, $delay));

$client = new Client([ 'handler' => $stack, 'timeout' => 5.0, ]);

  • 别写 new Client(['handler' => Middleware::retry(...)])),这会丢掉所有默认中间件,连重定向都失效
  • 如果你用了自定义 handler(比如 Swoole 的协程 handler),retry 中间件仍可用,但 $delay 的 usleep 行为可能被绕过,需确认底层是否支持
  • 日志调试时可加 on_stats 选项观察重试次数:'on_stats' => function (TransferStats $stats) { var_dump($stats->getEffectiveUri(), $stats->getTransferTime(), $stats->hasResponse()); }

重试真正难的不是代码几行,而是判断哪些错误值得重试、延迟设多少才既稳又快、以及确认底层 handler 是否尊重你的 delay 策略——这些不测一测线上流量,光看文档容易漏掉。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
swoole为什么能常驻内存
swoole为什么能常驻内存

swoole常驻内存的特性:1. 事件驱动模型减少内存消耗;2. 协程并行执行任务占用更少内存;3. 协程池预分配协程消除创建开销;4. 静态变量保留状态减少内存分配;5. 共享内存跨协程共享数据降低内存开销。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

302

2024.04.10

什么是中间件
什么是中间件

中间件是一种软件组件,充当不兼容组件之间的桥梁,提供额外服务,例如集成异构系统、提供常用服务、提高应用程序性能,以及简化应用程序开发。想了解更多中间件的相关内容,可以阅读本专题下面的文章。

180

2024.05.11

Golang 中间件开发与微服务架构
Golang 中间件开发与微服务架构

本专题系统讲解 Golang 在微服务架构中的中间件开发,包括日志处理、限流与熔断、认证与授权、服务监控、API 网关设计等常见中间件功能的实现。通过实战项目,帮助开发者理解如何使用 Go 编写高效、可扩展的中间件组件,并在微服务环境中进行灵活部署与管理。

222

2025.12.18

c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

241

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

641

2024.03.01

cookie
cookie

Cookie 是一种在用户计算机上存储小型文本文件的技术,用于在用户与网站进行交互时收集和存储有关用户的信息。当用户访问一个网站时,网站会将一个包含特定信息的 Cookie 文件发送到用户的浏览器,浏览器会将该 Cookie 存储在用户的计算机上。之后,当用户再次访问该网站时,浏览器会向服务器发送 Cookie,服务器可以根据 Cookie 中的信息来识别用户、跟踪用户行为等。

6441

2023.06.30

document.cookie获取不到怎么解决
document.cookie获取不到怎么解决

document.cookie获取不到的解决办法:1、浏览器的隐私设置;2、Same-origin policy;3、HTTPOnly Cookie;4、JavaScript代码错误;5、Cookie不存在或过期等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

355

2023.11.23

阻止所有cookie什么意思
阻止所有cookie什么意思

阻止所有cookie意味着在浏览器中禁止接受和存储网站发送的cookie。阻止所有cookie可能会影响许多网站的使用体验,因为许多网站使用cookie来提供个性化服务、存储用户信息或跟踪用户行为。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

418

2024.02.23

Golang处理数据库错误教程合集
Golang处理数据库错误教程合集

本专题整合了Golang数据库错误处理方法、技巧、管理策略相关内容,阅读专题下面的文章了解更多详细内容。

61

2026.02.06

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PHP课程
PHP课程

共137课时 | 11.3万人学习

JavaScript ES5基础线上课程教学
JavaScript ES5基础线上课程教学

共6课时 | 11.2万人学习

PHP新手语法线上课程教学
PHP新手语法线上课程教学

共13课时 | 0.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号