首页 > 后端开发 > Golang > 正文

如何在Golang中实现HTTP请求缓存

P粉602998670
发布: 2025-11-01 00:51:25
原创
309人浏览过
答案:通过实现自定义RoundTripper拦截请求,利用内存或外部存储缓存响应,结合HTTP头控制缓存有效性,可高效实现Golang HTTP缓存。

如何在golang中实现http请求缓存

在Golang中实现HTTP请求缓存,核心思路是拦截或封装HTTP客户端行为,在发送请求前检查是否存在有效缓存,若有则直接返回缓存结果,否则发起真实请求并保存响应。以下是一些实用且常见的实现方式。

使用中间件式缓存结构

通过自定义RoundTripper接口实现透明缓存,这种方式不会侵入业务代码,只需替换http.Client的传输层。

RoundTripper 接口定义了单个HTTP事务的执行逻辑,我们可以包装默认的http.Transport,在请求前后加入缓存判断。

示例:简易内存缓存实现

使用map存储响应,配合读写锁保证并发安全。

立即学习go语言免费学习笔记(深入)”;

type CacheTransport struct {
    transport http.RoundTripper
    cache     map[string]cachedResponse
    mu        sync.RWMutex
}

type cachedResponse struct {
    once      sync.Once
    resp      *http.Response
    bodyBytes []byte
}

func NewCacheTransport() *CacheTransport {
    return &CacheTransport{
        transport: http.DefaultTransport,
        cache:     make(map[string]cachedResponse),
    }
}

func (c *CacheTransport) RoundTrip(req *http.Request) (*http.Response, error) {
    key := req.Method + ":" + req.URL.String()

    c.mu.RLock()
    if cached, found := c.cache[key]; found {
        c.mu.RUnlock()
        // 返回缓存响应的副本,避免引用被修改
        respCopy := cached.resp.Clone(req.Context())
        body := bytes.NewReader(cached.bodyBytes)
        respCopy.Body = io.NopCloser(body)
        return respCopy, nil
    }
    c.mu.RUnlock()

    // 发起真实请求
    resp, err := c.transport.RoundTrip(req)
    if err != nil {
        return nil, err
    }

    // 读取响应体并缓存
    bodyBytes, err := io.ReadAll(resp.Body)
    if err != nil {
        resp.Body.Close()
        return nil, err
    }
    resp.Body.Close()

    respCopy := resp.Clone(req.Context())
    respCopy.Body = io.NopCloser(bytes.NewReader(bodyBytes))

    c.mu.Lock()
    c.cache[key] = cachedResponse{
        resp:      respCopy,
        bodyBytes: bodyBytes,
    }
    c.mu.Unlock()

    return respCopy, nil
}
登录后复制

使用方式:

存了个图
存了个图

视频图片解析/字幕/剪辑,视频高清保存/图片源图提取

存了个图17
查看详情 存了个图
client := &http.Client{
    Transport: NewCacheTransport(),
}

resp, err := client.Get("https://api.example.com/data")
登录后复制

控制缓存生命周期

实际应用中需考虑缓存过期机制。可基于HTTP标准头字段如 Cache-ControlExpiresLast-Modified 决定是否复用缓存。

  • 解析Cache-Control: max-age=60,表示最多缓存60秒
  • 若响应包含Expires,将其与当前时间比较
  • 结合ETagLast-Modified实现条件请求(如携带If-None-Match

可在CacheTransport中增强逻辑:缓存时记录时间,读取时验证有效性,过期后可选择删除或发起304 Not Modified验证。

选择合适的缓存存储

内存缓存适用于单机场景,但重启即丢失。根据需求可扩展为:

  • 本地磁盘:使用boltdbbadger持久化缓存数据
  • Redis:适合分布式服务共享缓存,支持TTL自动清理
  • LRU策略:使用container/list实现带容量限制的LRU缓存,防止内存无限增长

注意事项与优化建议

  • 仅对GET等幂等方法启用缓存
  • 注意请求头的影响(如Accept-Encoding),缓存键应包含关键头信息
  • 处理304响应时复用旧缓存,并更新过期时间
  • 避免缓存Set-Cookie等敏感头导致安全问题
  • 生产环境建议使用成熟库如gocache或集成fasthttp生态组件
基本上就这些。通过组合标准库接口和合理设计,Golang可以灵活实现高效HTTP缓存机制。

以上就是如何在Golang中实现HTTP请求缓存的详细内容,更多请关注php中文网其它相关文章!

最佳 Windows 性能的顶级免费优化软件
最佳 Windows 性能的顶级免费优化软件

每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。

下载
来源:php中文网
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
最新问题
开源免费商场系统广告
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板
关于我们 免责申明 意见反馈 讲师合作 广告合作 最新更新 English
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送
PHP中文网APP
随时随地碎片化学习
PHP中文网抖音号
发现有趣的

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