Flask 表单数据通过 SMTP 发送邮件的完整实现教程
技术百科
碧海醫心
发布时间:2026-01-01
浏览: 次 本文详解如何在 flask 应用中接收 html 表单数据,并使用 gmail smtp 安全地发送至指定邮箱,涵盖路由处理、邮件构造、tls 配置及常见失败原因排查。
在 Flask 中实现“表单提交 → 后端接收 → 邮件发送”功能时,一个常见误区是:仅定义了 send_email() 函数,却未在对应视图函数(如 contact())中显式调用它。从你的日志可见,POST /contact 请求已成功到达服务器(状态码 200),且控制台正确打印了表单字段值(如 Sara、[email protected] 等),说明数据已成功接收;但邮件未发出,根本原因正是 send_email() 未被触发。
以下是完整的、可直接运行的解决方案:
✅ 正确的 Flask 路由与邮件调用逻辑(main.py)
from flask import Flask, render_template, request, flash
import smtplib
from email.mime.text import MIMEText
from email.mime.multipart import MIMEMultipart
app = Flask(__name__)
app.secret_key = 'your-secret-key-here' # 用于 flash 消息(可选)
@app.route('/contact', methods=['GET', 'POST'])
def contact():
if request.method == 'POST':
# 1. 获取表单数据
name = request.form.get('name', '').strip()
email = request.form.get('email', '').strip()
phone = request.form.get('phone', '').strip()
message = request.form.get('message', '').strip()
# 2. 基础验证(防止空提交)
if not all([name, email, message]):
flash('Please fill in all required fields.', 'error')
return render_template('contact.html', msg_sent=False)
try:
# 3. 调用邮件发送函数(关键!此前缺失这一步)
send_email(name, email, phone, message)
return render_template('contact.html', msg_sent=True)
except Exception as e:
print(f"Email sending failed: {e}")
flash('Failed to send message. Please try again later.', 'error')
return render_template('contact.html', msg_sent=False)
# GET 请求:渲染初始页面
return render_template('contact.html', msg_sent=False)
def send_email(name, email, phone, message):
# ✅ 使用 Gmail App Password(非账户密码) + 启用两步验证后生成
smtp_server = 'smtp.gmail.com'
smtp_port = 587
smtp_username = 'your-verified-gmail@gmail.com' # 替换为你的邮箱
smtp_password = 'your-16-char-app-password' # 替换为 Google 生成的 App Password
from_email = smtp_username
to_email = 'recipient@example.com' # 替换为目标邮箱(可与 from_email 相同)
subject = f'New Contact Form Submission from {name}'
# ✅ 使用标准 MIME 格式(比纯字符串更可靠,避免编码/换行问题)
msg = MIMEMultipart()
msg['From'] = from_email
msg['To'] = to_email
msg['Subject'] = subject
body = f"""\
Name: {name}
Email: {email}
Phone: {phone}
Message:
{message}
"""
msg.attach(MIMEText(body, 'plain', 'utf-8'))
# ✅ 完整 SMTP 流程:连接 → TLS 加密 → 登录 → 发送
with smtplib.SMTP(smtp_server, smtp_port) as server:
server.starttls() # 启用 TLS
server.login(smtp_username, smtp_password)
server.send_message(msg) # 推荐使用 send_message() 替代 sendmail()? 关键注意事项
-
Gmail 设置必须正确:
- 开启 两步验证
- 在 App Passwords 页面 生成 16位应用专用密码(不是你的 Gmail 登录密码!)
- 确保 smtp_username 是你启用两步验证的 Gmail 账户(如 example@gmail.com)
HTML 表单需匹配后端字段名:
你的 contact.html 中 、不要忽略异常处理:
如示例中加入 try...except,能快速定位 SMTP 连接失败、认证错误或网络问题(例如防火墙拦截端口 587)。-

安全提醒:
切勿将 smtp_password 硬编码在代码中!生产环境应使用环境变量:import os smtp_password = os.getenv('GMAIL_APP_PASSWORD')并通过 .env 文件或系统环境变量管理。
✅ 验证是否成功
- 成功时:页面显示 “Successfully sent your message”,且收件箱收到格式清晰的邮件;
- 失败时:控制台打印详细错误(如 smtplib.SMTPAuthenticationError),据此调整凭据或网络配置。
遵循以上结构,你的 Flask 表单邮件功能即可稳定运行——核心在于 确保 send_email() 在 POST 请求处理逻辑中被主动调用,而非仅“存在”于文件中。
# ai
# 后端
# 可选
# 表单
# google
# 推荐使用
# 而非
# 可直接
# app
# word
# 防火墙
# 端口
# input
# go
# 路由
# 环境变量
# html
# 编码
# try
# 状态码
# protected
# 邮箱
# 表单提交
# 两步
# 收件箱
# flask
# 邮件发送
# 可与
相关栏目:
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
AI推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
SEO优化<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
技术百科<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
谷歌推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
百度推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
网络营销<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
案例网站<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
精选文章<?muma echo $count; ?>
】
相关推荐
- Windows10怎么卸载预装软件_Windows
- Win11怎么设置默认终端应用_Windows11
- php订单日志怎么记录评价_php记录订单评价日志
- 如何解决Windows字体显示模糊的问题?(Cle
- Win11摄像头无法使用怎么办_Win11相机隐私
- c++怎么使用std::filesystem遍历文
- PythonGIL机制理解_多线程限制解析【教程】
- C#怎么使用委托和事件 C# delegate与e
- php增删改查报错1054怎么办_字段名错误排查修
- 网站内页做seo排名怎么做?
- Linux如何安装Tomcat应用服务器_Linu
- Windows11怎样开启游戏模式_Windows
- Bpmn 2.0的XML文件怎么画流程图
- Win11任务栏怎么固定应用 Win11将软件图标
- 本地php环境打开php文件直接下载_浏览器解析p
- 如何在Golang中定义接口_抽象方法和多态实现
- Win11搜索栏无法输入_解决Win11开始菜单搜
- 如何使用Golang捕获并记录协程panic_保证
- MAC怎么一键隐藏桌面所有图标_MAC极简模式切换
- Python多进程教程_multiprocessi
- Win11怎么更改任务栏颜色_Windows11个
- Mac电脑进水了怎么办_MacBook进水后紧急处
- windows如何修改文件默认打开方式_windo
- C#怎么创建控制台应用 C# Console Ap
- Win10如何关闭安全中心所有通知 Win10禁用
- Win11怎么解压RAR文件 Win11自带解压功
- 如何使用Golang编写单元测试_创建Test函数
- 如何使用正则表达式批量替换重复的 *- 模式为固定
- Win11怎么自动隐藏任务栏_Win11全屏显示设
- 如何在Golang中实现RPC异步返回_Golan
- 如何使用Golang处理网络超时错误_Golang
- 如何使用Golang实现云原生应用弹性伸缩_自动应
- Windows10系统怎么查看显卡驱动_Win10
- Windows10电脑怎么设置虚拟内存_Win10
- 如何使用Golang log设置日志输出格式_Go
- 如何在 Python 中将 ISO 8601 时间
- 如何高效识别并拦截拼接式恶意域名 spam
- Linux如何安装Golang环境_Linux下G
- Python 模块的 __name__ 属性如何由
- Win11怎么设置默认邮件客户端 Win11修改M
- Python项目维护经验_长期演进说明【指导】
- C++如何使用std::transform批量处理
- MAC怎么解压RAR格式文件_MAC第三方解压工具
- 本地php环境出现502错误_nginx或apac
- Python解释执行模型_字节码流程说明【指导】
- Mac的“调度中心”与“空间”怎么用_Mac多桌面
- Python项目回滚策略_发布安全说明【指导】
- 如何在 Go 中正确初始化结构体中的 map 字段
- 如何正确访问 Laravel 模型或对象的属性而非
- Win11怎么设置任务栏图标大小_Windows1


QQ客服