url.Parse 从不返回 nil,失败时仍返回 *url.URL(部分字段为零值)和非 nil error;必须检查 err != nil 而非指针判空。
url.Parse 返回 nil 还是 error?它从不返回 nil,但会返回一个非空的 *url.URL 即使解析出错——比如 url.Parse("http://") 会返回一个 Host 为空的 URL 结构体,而错误只在第二个返回值中体现。必须检查 err != nil,不能靠判空指针。
url.Parse 成功时返回有效结构体 + nil 错误;失败时仍返回结构体(部分字段为零值)+ 非 nil 错误u, _ := url.Parse("htp://example.com") // 忽略 err,后续 u.Scheme == "" 导致逻辑崩溃u, err := url.Parse("https://api.example.com/v1?x=1&y=2")
if err != nil {
log.Fatal(err) // 或返回、返回默认值等
}url.Query 字段是已解码后的 url.Values,但直接访问 u.RawQuery 是原始未解码字符串。中文参数如 ?name=%E4%BD%A0%E5%A5%BD,用 u.Query().Get("name") 自动解码为 "你好";若手动处理 RawQuery,需调用 url.ParseQuery 并自行处理错误。
u.Query().Get("key"):自动解码,推荐用于常规取值u.Query().Set("k", "v"):自动编码,适合构造新 queryvalues, err := url.ParseQuery(u.RawQuery)
if err != nil { /* 处理 malformed query */ }RawQuery 直接做字符串切割或正则匹配——编码字符(如 %20)会被拆断url.URL 的 String() 方法是否总是生成合法 URL?不是。当 u.Opaque 非空(如 url.Parse("mailto:test@example.com"))时,String()

Host/Path 等字段,只输出 Opaque 内容;另外,如果 u.Scheme 为空但 u.Opaque 也为空,String() 可能返回不带 scheme 的路径,不符合标准 URL 格式。
u.Scheme != "" 且 u.Host != ""(对 http/https 而言)u2, err := url.Parse(u.String())
if err != nil || u2.Scheme == "" || u2.Host == "" { /* 不合法 */ }/,用 u.ResolveReference(&url.URL{Path: "/api"})
直接改 u.RawQuery 会破坏编码,应操作 u.Query() 返回的 url.Values,再赋回 u.RawQuery。
q := u.Query()
q.Set("t", strconv.FormatInt(time.Now().Unix(), 10))
u.RawQuery = q.Encode()u.RawQuery += "&t=..." —— 中文或特殊字符会乱码,且可能重复编码q.Del("token"),然后 u.RawQuery = q.Encode()
url.Values 是 map 类型,遍历时顺序不保证;如需稳定顺序,得自己排序后构建有些 URL 字段看似可写(比如 u.Path),但改完不调用 u.String() 或没重设 RawQuery 就直接使用,query 会丢失。net/url 包的字段不是全联动的,修改行为要严格按文档预期来。