如何在Golang中使用net/smtp发送邮件_构建并发送邮件内容

25次阅读

go中用net/smtp发邮件需手动构造RFC 5322格式邮件内容,支持纯文本、html及附件(需MIME多部分),通过smtp.SendMail连接认证后发送,注意Gmail等平台需app Passworddns验证。

如何在Golang中使用net/smtp发送邮件_构建并发送邮件内容

在 Go 中使用 net/smtp 发送邮件,核心是构造符合 SMTP 协议的邮件内容(即 RFC 5322 格式),再通过 SMTP 客户端连接并认证后发送。它不提供高级邮件构建能力(如 HTML、附件、内嵌图片等需手动处理 MIME),但足够轻量、可控。

构造标准邮件内容(纯文本或 HTML)

Go 标准库没有内置邮件构建器,需手动拼接邮件头与正文,用 rn 分隔,头部与正文间空一行。关键字段包括 FromToSubjectdateMIME-Version

例如发送纯文本邮件:

msg := []byte("To: recipient@example.comrn" +     "From: sender@example.comrn" +     "Subject: Hello from Gorn" +     "MIME-Version: 1.0rn" +     "Content-Type: text/plain; charset=utf-8rn" +     "rn" +     "This is a plain text email sent via net/smtp.")

若要发送 HTML 邮件,将 Content-Type 改为 text/html,并在正文中写 HTML 片段:

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

"Content-Type: text/html; charset=utf-8rn" + "rn" + "

Hello

This is HTML.

"

处理带附件的邮件(手动构造 MIME 多部分)

附件需使用 multipart/mixed 结构,边界(boundary)需唯一且不与正文冲突。建议用 mime/multipart 包辅助构造:

  • 创建 bytes.Buffer 作为输出缓冲区
  • multipart.NewWriter 写入 multipart 数据,设置随机 boundary
  • 先写文本/HTML 部分(multipart/alternative 可选),再写附件部分(application/octet-stream
  • 调用 w.Close() 自动写入 final boundary 和尾部

注意:附件文件需读取为字节,用 part.Write() 写入;文件名应通过 Content-Disposition 头指定,如:attachment; filename="report.pdf"

建立 SMTP 连接并发

使用 smtp.SendMail 最简,但仅支持 PLAIN 或 LOGIN 认证(不支持 OAuth2)。需提供:

  • SMTP 服务器地址(如 "smtp.gmail.com:587"
  • 身份认证信息(smtp.PlainAuth):用户名、密码、主机名(用于 HELO)、邮箱地址
  • 发件人地址(必须与认证账号一致,尤其 Gmail/outlook 等平台)
  • 收件人列表(字符串切片
  • 完整邮件字节数据(含头+正文)

示例:

auth := smtp.PlainAuth("", "user@gmail.com", "app-password", "smtp.gmail.com") err := smtp.SendMail("smtp.gmail.com:587", auth, "user@gmail.com", []string{"to@example.com"}, msg)

⚠️ 注意:Gmail 要求开启“两步验证”并生成 App Password;国内企业邮箱常需开启 SMTP 服务并检查端口(587 或 465)及 TLS 设置。

发送前校验与错误处理

常见失败点包括:

  • 邮箱格式错误(用 mail.ParseAddress 或正则预检)
  • SMTP 认证失败(用户名/密码错、App Password 过期、未开 SMTP)
  • 邮件头缺失或格式非法(如缺少 rn、Subject 含换行)
  • 被目标服务器拒收(SPF/DKIM/DMARC 验证失败,建议配置发信域名 DNS 记录)

建议对 SendMail 返回的 Error 做区分处理:网络错误可重试,认证失败需提示用户检查凭据,协议错误(如 5xx 响应)需解析 SMTP 错误码进一步诊断。

text=ZqhQzanResources