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; ?>
】
相关推荐
- C++中引用和指针有什么区别?(代码说明)
- Windows10怎样设置家长控制_Windows
- How to Properly Use NumPy
- c++如何使用std::bitset进行位图算法_
- 如何用::实现工具类方法调用_php静态工具类设计
- c++中的可变参数模板(variadic temp
- Win11无法识别耳机怎么办_解决Win11插耳机
- Win11怎么查看硬盘型号_Windows 11检
- Python装饰器复用技巧_通用能力解析【教程】
- Python包结构设计_大型项目组织解析【指导】
- Win10如何卸载微软拼音输入法 Win10只保留
- MAC如何隐藏文件夹及文件_MAC终端命令隐藏与第
- Windows10如何查看保存的WiFi密码_Wi
- Win11怎么关闭触摸屏_禁用Win11笔记本触摸
- php8.4如何实现队列任务_php8.4redi
- Win11相机打不开提示错误怎么修_相机权限开启与
- Python项目回滚策略_发布安全说明【指导】
- PythonGIL机制理解_多线程限制解析【教程】
- php会话怎么开启_session_start函数
- Python文件操作优化_大文件与流处理解析【教程
- Python 中将 ISO 8601 时间戳转换为
- 如何有效拦截拼接式恶意域名的垃圾信息
- Drupal 中 HTML 链接被双重转义导致渲染
- mac怎么分屏_MAC双屏显示与分屏操作技巧【指南
- 如何使用正则表达式提取以编号开头、后接多个注解的逻
- Win10电脑怎么设置网络名称_Windows10
- php485支持哪些操作系统_php485跨系统支
- 一文教你快速开通网站LOGO图
- Win11开机速度慢怎么优化_Win11系统启动加
- Win10如何关闭安全中心所有通知 Win10禁用
- SAX解析器是什么,它与DOM在处理大型XML文件
- Win11怎么打开旧版计算器_Win11恢复传统计
- 如何在 Go 中可靠地测试含 time.Time
- 如何使用Golang开发基础文件下载功能_Gola
- Win11怎么更改系统语言_Win11中文语言包下
- PythonPandas数据分析项目教程_时间序列
- PyTorch DDP 多进程训练在 Kaggle
- 微信短链接怎么还原php_用浏览器开发者工具抓包获
- php订单日志怎么导出excel_php导出订单日
- Win11怎么设置鼠标宏_Win11鼠标按键自定义
- 如何在Golang中验证模块完整性_Golangg
- Linux怎么实现内网穿透_Linux安装Frp客
- c++协程和线程的区别 c++异步编程模型对比【核
- C++如何获取CPU核心数?(std::threa
- Win11怎么硬盘分区 Win11新建磁盘分区详细
- c++怎么实现大文件的分块读写_c++ 文件指针s
- Windows10蓝屏SYSTEM_SERVICE
- Python与MongoDB NoSQL开发实战_
- Win11怎么关闭贴靠布局_Win11禁用窗口最大
- Win11怎么设置夜间模式_Windows11显示


QQ客服