0

0

Golang net/http 包实现服务器端 Cookie 管理

霞舞

霞舞

发布时间:2025-09-15 09:56:00

|

213人浏览过

|

来源于php中文网

原创

Golang net/http 包实现服务器端 Cookie 管理

本文详细介绍了如何在 Go 语言的 net/http 包中正确地从服务器端设置 HTTP Cookie。通过对比常见的错误用法(在请求对象上设置 Cookie)与正确实践(在响应写入器上设置 Cookie),文章重点阐述了 http.SetCookie 函数和 http.Cookie 结构体的应用,并提供了清晰的代码示例和关键字段的解释,帮助开发者高效管理 Web 应用中的用户会话和状态。

理解 HTTP Cookie 及其服务器端设置机制

http cookie 是一种由服务器发送到用户浏览器并存储在浏览器中的小型文本文件。它的主要作用是记录用户状态,例如用户登录信息、购物车内容、个性化设置等,以便在用户后续访问同一网站时,浏览器能将这些 cookie 随请求一同发送回服务器。

当服务器希望在客户端设置一个 Cookie 时,它会通过 HTTP 响应头中的 Set-Cookie 字段来指示浏览器。浏览器接收到这个响应头后,会解析其中的 Cookie 信息并将其存储起来。在后续对同一域名的请求中,浏览器会自动将这些存储的 Cookie 添加到请求头中的 Cookie 字段,发送给服务器。

在 Go 语言的 net/http 包中,正确地从服务器端设置 Cookie 是一个常见需求,但有时会因对 http.Request 和 http.ResponseWriter 职责的混淆而导致错误。

Go net/http 中设置 Cookie 的常见误区

许多初学者可能会尝试通过 http.Request 对象来设置 Cookie,例如使用 req.AddCookie() 方法。然而,这是一个常见的误解。http.Request 对象代表的是客户端发送到服务器的请求,其上的 AddCookie() 方法主要用于在 Go 客户端发起 HTTP 请求时,向 该请求 中添加 Cookie,以便将它们发送到 另一个服务器

例如:

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

// 这是一个客户端请求的例子,与服务器端设置Cookie无关
client := &http.Client{}
req, _ := http.NewRequest("GET", "http://example.com", nil)
cookie := &http.Cookie{Name: "auth", Value: "token123"}
req.AddCookie(cookie) // 将Cookie添加到要发送出去的请求中
resp, _ := client.Do(req)
// ... 处理响应

当我们需要服务器向浏览器发送 Cookie 时,我们操作的是 http.ResponseWriter 对象,因为它负责构建并发送 HTTP 响应。

正确地从服务器端设置 Cookie

在 Go 语言中,要从服务器端设置 Cookie,需要使用 http.ResponseWriter 接口配合 http.SetCookie 函数。这个函数会负责生成正确的 Set-Cookie HTTP 响应头,并将其添加到发送给客户端的响应中。

宣小二
宣小二

宣小二:媒体发稿平台,自媒体发稿平台,短视频矩阵发布平台,基于AI驱动的企业自助式投放平台。

下载

核心步骤如下:

  1. 创建 http.Cookie 实例: 定义一个 http.Cookie 结构体,填充其字段,如 Name、Value、Expires、Path、Domain、HttpOnly 和 Secure 等。
  2. 调用 http.SetCookie: 将 http.ResponseWriter 和创建的 http.Cookie 实例作为参数传递给 http.SetCookie 函数。

示例代码:实现服务器端 Cookie 设置

下面是一个完整的 Go Web 服务器示例,演示了如何正确地在响应中设置 Cookie:

package main

import (
    "fmt"
    "net/http"
    "time"
)

// setCookieHandler 负责在 HTTP 响应中设置一个 Cookie
func setCookieHandler(w http.ResponseWriter, r *http.Request) {
    // 1. 定义 Cookie 的过期时间
    //    这里设置为当前时间起 24 小时后过期
    expiration := time.Now().Add(24 * time.Hour)

    // 2. 创建一个 http.Cookie 结构体实例
    //    此 Cookie 将在客户端浏览器中存储,并随后续请求发送回服务器。
    cookie := &http.Cookie{
        Name:    "user_session",       // Cookie 的名称,用于标识
        Value:   "abcdef1234567890",   // Cookie 的值,通常是会话ID或令牌
        Path:    "/",                  // Cookie 的作用路径,"/" 表示对整个网站有效
        // Domain:  "localhost",        // 可选:指定 Cookie 所属的域名。
        //          如果未设置,默认为当前请求的域名。
        //          注意:不能设置为其他域名,只能是当前域名或其子域名。
        Expires: expiration,           // 设置 Cookie 的绝对过期时间
        // MaxAge:  86400,              // 可选:设置 Cookie 的最大存活时间(秒)。
        //          如果同时设置 Expires 和 MaxAge,MaxAge 优先级更高(在某些浏览器中)。
        HttpOnly: true,                // 设为 true 可防止客户端脚本(如 JavaScript)访问 Cookie,增加安全性
        Secure:   false,               // 设为 true 仅在 HTTPS 连接中发送 Cookie。
        //          生产环境强烈建议设置为 true。
        SameSite: http.SameSiteLax,    // 增加 CSRF 保护,推荐设置。
        //          可选值:http.SameSiteDefaultMode, http.SameSiteLaxMode, http.SameSiteStrictMode, http.SameSiteNoneMode
    }

    // 3. 使用 http.SetCookie 函数将 Cookie 添加到 HTTP 响应头中
    //    这个函数会生成一个 "Set-Cookie" 响应头,发送给客户端浏览器。
    http.SetCookie(w, cookie)

    fmt.Fprintf(w, "Cookie 'user_session' 已成功设置。请检查浏览器开发者工具中的Cookie。")
}

// main 函数启动 HTTP 服务器并注册处理函数
func main() {
    http.HandleFunc("/set-cookie", setCookieHandler) // 注册设置 Cookie 的路由
    fmt.Println("服务器正在监听 :8080,请访问 http://localhost:8080/set-cookie")
    err := http.ListenAndServe(":8080", nil) // 启动服务器
    if err != nil {
        fmt.Printf("服务器启动失败: %v\n", err)
    }
}

运行上述代码后,访问 http://localhost:8080/set-cookie,你将在浏览器开发者工具的网络请求响应头中看到 Set-Cookie 字段,并且在浏览器存储中找到名为 user_session 的 Cookie。

http.Cookie 结构体字段详解

http.Cookie 结构体定义了 Cookie 的各种属性:

  • Name (string): Cookie 的名称。
  • Value (string): Cookie 的值。
  • Path (string): Cookie 的有效路径。只有当请求的 URL 路径匹配或包含此路径时,浏览器才会发送此 Cookie。"/" 表示对整个网站有效。
  • Domain (string): Cookie 的有效域名。只有当请求的域名匹配此域名时,浏览器才会发送此 Cookie。如果未设置,默认为当前请求的域名。注意: 不能设置为其他域名,只能是当前域名或其子域名。
  • Expires (time.Time): Cookie 的过期时间(绝对时间)。一旦到达此时间,Cookie 将被浏览器删除。如果未设置或设置为零值,则 Cookie 为会话 Cookie,在浏览器关闭时失效。
  • MaxAge (int): Cookie 的最大存活时间(秒)。从设置时开始计算。如果 MaxAge 为 0,Cookie 立即过期;如果为负数,Cookie 将被删除。当 Expires 和 MaxAge 同时设置时,MaxAge 在某些浏览器中可能具有更高优先级。
  • HttpOnly (bool): 如果设置为 true,则客户端脚本(如 JavaScript)无法通过 document.cookie 等 API 访问此 Cookie,有助于防止跨站脚本 (XSS) 攻击。
  • Secure (bool): 如果设置为 true,则此 Cookie 仅在 HTTPS 连接中发送。强烈建议在生产环境中将此属性设置为 true,以防止 Cookie 在不安全的 HTTP 连接中被窃听。
  • SameSite (http.SameSite): 用于控制 Cookie 在跨站点请求中的发送行为,有助于防止跨站请求伪造 (CSRF) 攻击。
    • http.SameSiteDefaultMode: 浏览器默认行为。
    • http.SameSiteLax: 宽松模式,在顶级导航和 GET 请求中发送。
    • http.SameSiteStrictMode: 严格模式,仅在同站点请求中发送。
    • http.SameSiteNoneMode: 总是发送,但必须同时设置 Secure 为 true。

注意事项与最佳实践

  1. 安全性优先:
    • Secure 属性: 在生产环境中,务必将 Secure 属性设置为 true,确保 Cookie 仅通过 HTTPS 连接发送,防止中间人攻击窃取 Cookie。
    • HttpOnly 属性: 对于包含敏感信息(如会话 ID)的 Cookie,应将 HttpOnly 设置为 true,以防止 XSS 攻击获取 Cookie。
    • SameSite 属性: 推荐使用 SameSiteLax 或 SameSiteStrictMode 来增强 CSRF 保护。
  2. 作用域管理:
    • Path 和 Domain: 精确设置 Path 和 Domain 可以限制 Cookie 的发送范围,避免不必要的 Cookie 传输,提高效率和安全性。
  3. 避免存储敏感信息: 避免在 Cookie 中直接存储密码、信用卡号等高度敏感的用户信息。如果必须存储,请确保对其进行加密或使用安全的会话 ID 引用服务器端存储的数据。
  4. 过期时间: 合理设置 Expires 或 MaxAge。对于会话 Cookie,不设置过期时间即可(浏览器关闭即失效);对于需要持久化的 Cookie,应设置一个合理的过期时间。
  5. Cookie 大小限制: 浏览器对单个 Cookie 的大小和每个域名的 Cookie 数量都有限制(通常单个 Cookie 4KB,每个域名 20-50 个)。避免存储过大的数据。

总结

在 Go 语言中,通过 net/http 包从服务器端设置 Cookie 的核心在于理解 http.ResponseWriter 的作用,并正确使用 http.SetCookie 函数和 http.Cookie 结构体。避免在 http.Request 上操作 Cookie 是关键。遵循 Secure、HttpOnly 和 SameSite 等安全属性的最佳实践,能够有效提高 Web 应用程序的安全性,并为用户提供更稳定可靠的体验。

相关专题

更多
js获取数组长度的方法
js获取数组长度的方法

在js中,可以利用array对象的length属性来获取数组长度,该属性可设置或返回数组中元素的数目,只需要使用“array.length”语句即可返回表示数组对象的元素个数的数值,也就是长度值。php中文网还提供JavaScript数组的相关下载、相关课程等内容,供大家免费下载使用。

556

2023.06.20

js刷新当前页面
js刷新当前页面

js刷新当前页面的方法:1、reload方法,该方法强迫浏览器刷新当前页面,语法为“location.reload([bForceGet]) ”;2、replace方法,该方法通过指定URL替换当前缓存在历史里(客户端)的项目,因此当使用replace方法之后,不能通过“前进”和“后退”来访问已经被替换的URL,语法为“location.replace(URL) ”。php中文网为大家带来了js刷新当前页面的相关知识、以及相关文章等内容

374

2023.07.04

js四舍五入
js四舍五入

js四舍五入的方法:1、tofixed方法,可把 Number 四舍五入为指定小数位数的数字;2、round() 方法,可把一个数字舍入为最接近的整数。php中文网为大家带来了js四舍五入的相关知识、以及相关文章等内容

734

2023.07.04

js删除节点的方法
js删除节点的方法

js删除节点的方法有:1、removeChild()方法,用于从父节点中移除指定的子节点,它需要两个参数,第一个参数是要删除的子节点,第二个参数是父节点;2、parentNode.removeChild()方法,可以直接通过父节点调用来删除子节点;3、remove()方法,可以直接删除节点,而无需指定父节点;4、innerHTML属性,用于删除节点的内容。

477

2023.09.01

JavaScript转义字符
JavaScript转义字符

JavaScript中的转义字符是反斜杠和引号,可以在字符串中表示特殊字符或改变字符的含义。本专题为大家提供转义字符相关的文章、下载、课程内容,供大家免费下载体验。

434

2023.09.04

js生成随机数的方法
js生成随机数的方法

js生成随机数的方法有:1、使用random函数生成0-1之间的随机数;2、使用random函数和特定范围来生成随机整数;3、使用random函数和round函数生成0-99之间的随机整数;4、使用random函数和其他函数生成更复杂的随机数;5、使用random函数和其他函数生成范围内的随机小数;6、使用random函数和其他函数生成范围内的随机整数或小数。

1011

2023.09.04

如何启用JavaScript
如何启用JavaScript

JavaScript启用方法有内联脚本、内部脚本、外部脚本和异步加载。详细介绍:1、内联脚本是将JavaScript代码直接嵌入到HTML标签中;2、内部脚本是将JavaScript代码放置在HTML文件的`<script>`标签中;3、外部脚本是将JavaScript代码放置在一个独立的文件;4、外部脚本是将JavaScript代码放置在一个独立的文件。

658

2023.09.12

Js中Symbol类详解
Js中Symbol类详解

javascript中的Symbol数据类型是一种基本数据类型,用于表示独一无二的值。Symbol的特点:1、独一无二,每个Symbol值都是唯一的,不会与其他任何值相等;2、不可变性,Symbol值一旦创建,就不能修改或者重新赋值;3、隐藏性,Symbol值不会被隐式转换为其他类型;4、无法枚举,Symbol值作为对象的属性名时,默认是不可枚举的。

553

2023.09.20

Java JVM 原理与性能调优实战
Java JVM 原理与性能调优实战

本专题系统讲解 Java 虚拟机(JVM)的核心工作原理与性能调优方法,包括 JVM 内存结构、对象创建与回收流程、垃圾回收器(Serial、CMS、G1、ZGC)对比分析、常见内存泄漏与性能瓶颈排查,以及 JVM 参数调优与监控工具(jstat、jmap、jvisualvm)的实战使用。通过真实案例,帮助学习者掌握 Java 应用在生产环境中的性能分析与优化能力。

19

2026.01.20

热门下载

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

精品课程

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

共58课时 | 3.9万人学习

TypeScript 教程
TypeScript 教程

共19课时 | 2.3万人学习

Bootstrap 5教程
Bootstrap 5教程

共46课时 | 2.9万人学习

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

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