登录
推荐 文章 Go 技术 课程 下载 专题 AI
首页 >  文章 >  python教程

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

来源:17golang原创

时间:2026-06-27 17:57:42 478浏览 收藏

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删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>