在 go 中使用标准库 `net/smtp` 发送邮件时,`client.mail()` 仅接受纯邮箱地址作为参数,而显示给收件人的“发件人名称”需通过邮件头(`from` 字段)设置为 `"姓名
Go 的标准 net/smtp 包要求严格区分SMTP 协议层的发件人身份(用于认证与投递校验)和邮件内容层的显示发件人(用于收件人界面展示)。若将 "Sandy Sender
邮箱地址(如
✅ 正确做法是 “双轨设置”:
以下是一个完整、可运行的标准库示例(使用 strings.Reader 构造原始邮件):
package main
import (
"fmt"
"net/smtp"
"strings"
)
func main() {
// SMTP 配置
auth := smtp.PlainAuth("", "user@example.com", "app-password", "smtp.example.com")
// 构造符合 RFC 5322 的原始邮件(注意:From 头含名称)
msg := strings.NewReader(`From: Sandy Sender
To: recipient@example.com
Subject: Hello from Go!
MIME-Version: 1.0
Content-Type: text/plain; charset="utf-8"
This is the body of the message.
`)
// 关键:Client.Mail() 只传纯邮箱地址(无名称!)
err := smtp.SendMail(
"smtp.example.com:587",
auth,
"user@example.com", // ← 必须是纯邮箱(与 From 头中的地址一致)
[]string{"recipient@example.com"},
msg,
)
if err != nil {
panic(fmt.Sprintf("failed to send email: %v", err))
}
fmt.Println("Email sent successfully!")
} ⚠️ 注意事项:
m := gomail.NewMessage()
m.SetAddressHeader("From", "user@example.com", "Sandy Sender") // 自动组装为 "Sandy Sender "
m.SetAddressHeader("To", "recipient@example.com", "")
m.SetHeader("Subject", "Hello!")
m.SetBody("text/plain", "This is the body.")
// ... 发送逻辑 gomail 内部已封装上述分离逻辑,开发者无需手动拼接原始邮件,显著降低出错风险。
总结:Go 原生 SMTP 发送邮件时,“显示名称”与“协议发件人”必须解耦处理——前者写入邮件头 From 字段,后者单独传入 Client.Mail()。理解这一分层设计,是实现专业邮件发送的关键基础。