Go语言实现邮件发送功能_Golang通知系统实战

9次阅读

常见原因是SMTP服务端拒绝未认证或未加密连接,需用PlainAuth显式认证、选587/465端口qq邮箱用授权码、Gmail用应用专用密码。

Go语言实现邮件发送功能_Golang通知系统实战

为什么 net/smtp 发不出邮件却没报错?

常见现象是调用 smtp.SendMail 返回 nil 错误,但收件人根本没收到——这通常不是代码问题,而是 SMTP 服务端拒绝了未认证或未加密的连接。Gmail、outlook、QQ 邮箱等主流服务商默认禁用不带身份验证的明文 SMTP(端口 25),也拒绝未启用 TLS 的连接。

  • 必须使用 auth := smtp.PlainAuth("", user, password, host) 显式传入认证凭据
  • 优先选端口 587(STARTTLS)或 465(SMTPS),避免用 25
  • 若用 QQ 邮箱,user 必须是完整邮箱地址(如 "xxx@qq.com"),密码需填「SMTP 授权码」而非登录密码
  • Gmail 要求开启「两步验证」并生成应用专用密码;否则会返回 535 5.7.8 Username and Password not accepted

如何构造带 html 内容和附件的 multipart 邮件?

go 标准库没有开箱即用的 MIME 构建器,得手动拼装 multipart/mixedmultipart/alternative 结构。关键在于边界(boundary)一致、头部顺序正确、正文编码合规。

  • mime/multipart 创建 Writer,调用 w.WriteField("To", "...") 设置基础头字段
  • 先写 multipart/alternative 子部分(含纯文本 text/plain 和 HTML text/html),再写附件部分
  • HTML 正文必须用 base64 编码,并声明 Content-Transfer-Encoding: base64
  • 附件文件名含中文时,要用 mime.BEncoding.Encode("UTF-8", "简历.pdf") 编码,否则 Outlook 会显示乱码
body := &bytes.Buffer{} writer := multipart.NewWriter(body) writer.SetBoundary("myboundary123")  // HTML 部分 part, _ := writer.CreatePart(map[string][]string{ 	"Content-Type": {"text/html; charset=UTF-8"}, 	"Content-Transfer-Encoding": {"base64"}, }) base64.NewEncoder(base64.StdEncoding, part).Write([]byte(`

你好

`)) // 附件 part, _ = writer.CreatePart(map[string][]string{ "Content-Type": {"application/pdf"}, "Content-Disposition": {`attachment; filename*=UTF-8''%E7%AE%80%E5%8E%86.pdf`}, }) io.Copy(part, file) writer.Close()

发送失败时怎么快速定位是网络、认证还是内容问题?

错误信息常被忽略,但 smtp.SendMail 返回的 Error 值里包含原始 SMTP 状态码和服务器提示,是第一手线索。

  • 捕获错误后,先检查是否为 *textproto.Error 类型:if e, ok := err.(*textproto.Error); ok { fmt.Println(e.Code, e.Msg) }
  • 421 表示服务不可用(DNS 解析失败或目标主机拒连);454 是 TLS 启动失败;535 是认证失败;554 多为内容被判定为垃圾邮件
  • 本地测试可用 telnet smtp.qq.com 587 手动走一遍 EHLO → AUTH → MAIL FROM 流程,确认基础连通性
  • 若日志中反复出现 550 Mail from ... not allowed,说明发信域名未配置 SPF 或 DKIM,需在 DNS 添加记录

生产环境要不要自己搭 SMTP 中继?

直接调用第三方 SMTP(如 SendGrid、Mailgun、腾讯云 SES)比自建更可靠,尤其对通知类高频小邮件。

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

  • 自建 Postfix + OpenDKIM 虽可控,但维护成本高:IP 被拉黑、HELO 域名不匹配、反向 DNS 缺失都会导致投递失败
  • 云服务提供 Webhook 回调、发送统计、退订管理,还能自动降级(如邮件失败时 fallback 到企业微信
  • 若必须自建,至少保证:出站 IP 有固定公网地址、PTR 记录与 HELO 域名一致、每天发信量控制在 100 封以内防进垃圾箱
  • Go 客户端对接云服务时,别硬编码 API Key,用 os.Getenv("MAILGUN_API_KEY") + .env 文件管理

实际最难的不是发出去,是让收件方的 Gmail 不把它扔进推广栏——这取决于发信域名信誉、内容文本比例、链接数量,还有你有没有在 HTML 里偷偷加 display:none 的关键词。

text=ZqhQzanResources