当前位置:首页 > 文章列表 > 文章 > python教程 > Python requests 请求总是卡住?timeout、重试和错误处理配方

Python requests 请求总是卡住?timeout、重试和错误处理配方

来源:17golang原创 2026-06-27 17:57:42 0浏览 收藏

Python 项目里用 requests 调接口很顺手,但线上最常见的问题也很直接:某个外部接口慢了,调用方线程一直等,页面转圈、任务堆积、日志里只剩下“还没返回”。这篇文章按配方卡片的方式处理一个具体问题:给所有外部 HTTP 请求加上合理的 timeout、可控的重试和清晰的错误边界。

目录
  • 问题:请求为什么会一直卡住
  • 最小配方:每个外部请求都写 timeout
  • 关键代码:连接超时和读取超时分开设
  • 变体一:Session + Retry 统一处理重试
  • 变体二:封装业务请求函数
  • 常见坑和检查清单
  • 完整片段

问题:请求为什么会一直卡住

很多示例代码会写成 requests.get(url)。在本地测试时它看起来没有问题,因为目标接口通常很快返回;到了线上,DNS、网络连接、代理、对方服务排队、响应体过大,都可能让调用方等待很久。真正危险的点不是“请求失败”,而是“请求迟迟没有失败”。

建议把外部调用当成不可靠依赖来处理:每次请求必须有时间上限,失败要能分类,重试必须有次数和退避等待,业务层要拿到明确的错误。

最小配方:每个外部请求都写 timeout

最小可用写法是给 timeout 传一个二元组:第一个值控制建立连接的等待时间,第二个值控制读取响应的等待时间。这样既能快速发现网络不可达,也不会因为对方慢慢吐数据而无限等待。

Python requests timeout 分成连接超时和读取超时,再检查状态并返回 JSON 的流程图
图 1:最小配方把一次请求拆成连接、读取、状态检查和解析四个明确阶段。
import requests

resp = requests.get(
    "https://api.example.com/orders",
    params={"page": 1},
    timeout=(3, 10),
)
resp.raise_for_status()
data = resp.json()

这里的 (3, 10) 不是固定标准,而是一个起点:连接 3 秒仍然无法建立,通常可以认为网络或目标地址有问题;连接成功后 10 秒还没有读完响应,就要看接口类型、响应体大小和业务容忍度。

关键代码:连接超时和读取超时分开设

timeout=10 也能用,但它把连接和读取混在一个数里,不利于定位问题。实际排查时,连接超时通常指向网络、域名、代理或端口;读取超时更常见于对方服务慢、SQL 慢、队列排队或返回体太大。

try:
    resp = requests.get(
        "https://api.example.com/profile",
        timeout=(2, 8),
    )
    resp.raise_for_status()
except requests.ConnectTimeout as err:
    raise RuntimeError("connect timeout") from err
except requests.ReadTimeout as err:
    raise RuntimeError("read timeout") from err

如果你只想在业务层展示统一提示,可以捕获 requests.Timeout;如果你正在做线上问题定位,建议先把连接和读取分开记录到日志里。

变体一:Session + Retry 统一处理重试

单次请求设置 timeout 只能避免一直等待,不能处理偶发的 429502503。这类状态码常见于限流、网关抖动或服务短暂不可用,可以给幂等请求增加少量重试。

Python requests Session 和 Retry 对 429 与 5xx 状态码进行退避重试的流程图
图 2:重试逻辑放在 Session 层,业务代码只关心最终成功或明确失败。
from requests.adapters import HTTPAdapter
from urllib3.util.retry import Retry
import requests

retry = Retry(
    total=3,
    connect=3,
    read=2,
    backoff_factor=0.5,
    status_forcelist=(429, 500, 502, 503, 504),
    allowed_methods=("GET", "HEAD", "OPTIONS"),
)

session = requests.Session()
adapter = HTTPAdapter(max_retries=retry)
session.mount("http://", adapter)
session.mount("https://", adapter)

这段配置的重点有三个:只重试少数明确状态码;只默认重试幂等方法;用 backoff_factor 做退避等待,避免失败后立刻连续打满对方服务。

变体二:封装业务请求函数

如果项目里每个地方都直接调用 requests.get,超时、重试、日志、状态码检查很容易写散。更稳的做法是封装一个小函数,让业务层拿到结构化结果,或者拿到明确异常。

def fetch_json(url: str, *, params: dict | None = None) -> dict:
    try:
        resp = session.get(url, params=params, timeout=(3, 10))
        resp.raise_for_status()
        return resp.json()
    except requests.Timeout as err:
        raise RuntimeError("request timeout") from err
    except requests.HTTPError as err:
        status_code = err.response.status_code if err.response else "unknown"
        raise RuntimeError(f"bad status: {status_code}") from err
    except requests.RequestException as err:
        raise RuntimeError("request failed") from err

封装后,调用方不需要关心底层库的各种异常类型。对外部接口较多的系统,还可以在这里统一加入请求 ID、耗时日志、响应体大小限制和错误上报。

常见坑和检查清单

坑 1:只写重试,不写 timeout

没有时间上限的请求会先卡住,重试逻辑根本没有机会触发。顺序应该是先限制每次请求的最长等待,再决定是否重试。

坑 2:对 POST 默认重试

创建订单、扣库存、扣款这类请求通常不是天然幂等。除非服务端有幂等键,否则不要随意重试会改变数据的请求。

坑 3:timeout 设得过长

把超时设成 60 秒看似“更稳”,实际上会拖住工作线程。接口面向用户时,通常应该更短;后台批处理可以稍长,但也要有总耗时预算。

坑 4:吞掉原始异常

业务层可以返回统一错误,但日志里要保留原始异常、URL 域名、状态码、耗时和重试次数。否则问题发生时很难判断是网络、网关还是对方服务慢。

完整片段

from requests.adapters import HTTPAdapter
from urllib3.util.retry import Retry
import requests


def build_session() -> requests.Session:
    retry = Retry(
        total=3,
        connect=3,
        read=2,
        backoff_factor=0.5,
        status_forcelist=(429, 500, 502, 503, 504),
        allowed_methods=("GET", "HEAD", "OPTIONS"),
    )
    adapter = HTTPAdapter(max_retries=retry)

    session = requests.Session()
    session.mount("http://", adapter)
    session.mount("https://", adapter)
    return session


session = build_session()


def fetch_json(url: str, *, params: dict | None = None) -> dict:
    try:
        resp = session.get(url, params=params, timeout=(3, 10))
        resp.raise_for_status()
        return resp.json()
    except requests.Timeout as err:
        raise RuntimeError("request timeout") from err
    except requests.HTTPError as err:
        status_code = err.response.status_code if err.response else "unknown"
        raise RuntimeError(f"bad status: {status_code}") from err
    except requests.RequestException as err:
        raise RuntimeError("request failed") from err


orders = fetch_json("https://api.example.com/orders", params={"page": 1})

总结一下:requests 的稳定用法不是多写几行配置,而是把外部依赖的边界讲清楚。每次请求设置连接和读取超时;对少量可重试状态做有限重试;在封装函数里统一状态检查和错误分类。这样接口慢的时候,系统会明确失败,而不是悄悄卡住。

版本声明
本文转载于:17golang原创 如有侵犯,请联系study_golang@163.com删除
接口返回的数据和数据库不一致怎么办?按数据生命周期排查接口返回的数据和数据库不一致怎么办?按数据生命周期排查
上一篇
接口返回的数据和数据库不一致怎么办?按数据生命周期排查
MySQL 云上架构怎么选:单库、读写分离和分库分表的决策指南
下一篇
MySQL 云上架构怎么选:单库、读写分离和分库分表的决策指南
查看更多
最新文章
查看更多
课程推荐
  • 前端进阶之JavaScript设计模式
    前端进阶之JavaScript设计模式
    设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
    543次学习
  • GO语言核心编程课程
    GO语言核心编程课程
    本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
    516次学习
  • 简单聊聊mysql8与网络通信
    简单聊聊mysql8与网络通信
    如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
    500次学习
  • JavaScript正则表达式基础与实战
    JavaScript正则表达式基础与实战
    在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
    487次学习
  • 从零制作响应式网站—Grid布局
    从零制作响应式网站—Grid布局
    本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
    485次学习
查看更多
AI推荐
  • ljg-skills -
    ljg-skills
    ljg-skills 是李继刚开源的 AI 技能与提示词集合,面向大模型使用者整理了一批可复用的 prompt、角色设定和任务技能模板,适合用于学习提示词设计、搭建个人 AI 工作流和沉淀团队常用智能体能力。
    2588次使用
  • MELO音乐 - AI 音乐生成平台,支持多模态创作能力
    MELO音乐
    MELO音乐是一站式AI视频与音乐制作助手,对标suno, udio的高品质体验。提供伴奏生成、原创写词、无损导出、哼唱识曲、混音变声等全套音频与短视频编辑工具。无论是流行Kpop、电音说唱、民谣古风、摇滚儿歌还是商用轻音乐,MELO为你免费谱曲,轻松做同款!
    2394次使用
  • UniScribe - AI 免费在线音视频转文字平台
    UniScribe
    UniScribe 是一款 AI 音视频转文字与内容整理工具,支持上传音频、视频文件或粘贴 YouTube 链接,自动生成转写文本、摘要、思维导图和关键问题,并支持多格式导出,适合会议记录、课程学习、访谈整理和内容创作复盘。
    2336次使用
  • 剧云 - 免费 AI 智能中文剧本创作平台
    剧云
    剧云是专业中文剧本创作平台,安全稳定运行十余年,集成AI编剧、剧本医生审核、人物小传、剧情关系图、大纲编写、多人协作、Word导入导出、版权管控功能,数据安全防护,轻松高效创作剧本。
    2549次使用
  • 万象有声 - AI 一站式有声内容创作平台
    万象有声
    万象有声,一个专为有声创作者打造的新一代智能有声内容创作平台。平台提供专业的智能拆章、智能画本编辑、AI配音、AI生成音效、后期制作、智能对轨、智能审听等有声创作全流程工具,可以帮助创作者高效、低成本创作出引人入胜的有声作品。立即体验,让有声书制作更简单!
    2525次使用
微信登录更方便
  • 密码登录
  • 注册账号
登录即同意 用户协议隐私政策
返回登录
  • 重置密码