当前位置:首页 > 文章列表 > 文章 > python教程 > Python发送邮件教程:smtplib使用全解析

Python发送邮件教程:smtplib使用全解析

2025-08-06 15:01:53 0浏览 收藏

想知道如何用Python轻松玩转邮件发送吗?这篇教程将带你从入门到精通,手把手教你使用`smtplib`模块发送邮件。你将学会如何配置邮箱信息,解决常见的发送失败问题,例如授权码错误、服务器配置问题等。更进一步,还将学习如何利用`email`模块的MIME组件,发送带附件和HTML内容的精美邮件。同时,我们还会探讨大批量发送邮件的注意事项,教你如何避免被邮箱服务商封禁,并推荐专业的邮件服务,如SendGrid、阿里云邮件推送,助你提升邮件送达率。无论你是新手还是有一定经验的开发者,都能从中获益,掌握稳定高效的Python邮件发送技巧。

邮件发送失败常见原因包括:未使用邮箱授权码而直接使用登录密码;SMTP服务器地址或端口配置错误(如QQ邮箱应使用smtp.qq.com:465用于SSL);网络或防火墙限制导致无法连接;邮件内容被识别为垃圾邮件;邮箱地址拼写错误。2. 发送带附件或HTML内容的邮件需使用email模块的MIME组件:HTML内容通过MIMEText(content, 'html', 'utf-8')实现;附件需读取为二进制数据,用MIMEBase封装并Base64编码,再通过Content-Disposition头设置为附件。3. 大批量发送邮件时应避免使用个人邮箱SMTP服务,因其有频率和数量限制,易被封禁;建议采用专业邮件服务如SendGrid、阿里云邮件推送等以提高送达率;若自行发送,需控制频率、添加异常重试机制、记录日志并安全存储授权码(如使用环境变量)。使用Python发送电子邮件的核心是smtplib与email模块协同工作,遵循此流程可实现稳定高效的邮件发送功能。

Python如何发送电子邮件?smtplib模块完整流程

使用Python发送电子邮件,核心在于利用其内置的smtplib模块来与邮件服务器进行通信,以及email模块来构建符合MIME标准的邮件内容。简单来说,就是通过代码模拟一个邮件客户端的行为:连接到服务器、登录、然后把准备好的邮件内容“投递”出去。这听起来有点像在邮局寄信,smtplib是邮递员,email模块则帮你把信封、信纸、照片(附件)都整理得妥妥当当。

解决方案

要用Python发送电子邮件,我们通常会经历以下几个步骤,这基本上是一个标准流程:

  1. 导入必要的模块smtplib负责连接和发送,email.mime.text.MIMEText用于纯文本或HTML内容,email.mime.multipart.MIMEMultipart用于组合多部分邮件(如文本+附件),而email.mime.base.MIMEBaseemail.encoders则用于处理附件。
  2. 配置邮件信息:包括发件人邮箱、收件人邮箱、邮件主题、邮件正文,以及SMTP服务器地址和端口。特别要注意的是,很多主流邮箱服务(如QQ邮箱、163邮箱、Gmail等)出于安全考虑,不再允许你直接使用登录密码进行SMTP认证,而是需要生成一个“授权码”或“应用专用密码”。这往往是新手最容易踩的坑。
  3. 构建邮件内容:使用MIMEMultipart对象作为邮件的“容器”,然后将文本、HTML内容(如果需要)或附件作为子部分添加到这个容器中。
  4. 建立SMTP连接:根据邮件服务器的要求,选择smtplib.SMTP_SSL(推荐,端口通常是465)进行SSL加密连接,或者smtplib.SMTP配合starttls()(端口通常是587)进行TLS加密。
  5. 登录邮件服务器:使用发件人邮箱和前面提到的授权码进行身份验证。
  6. 发送邮件:调用send_message()sendmail()方法发送邮件。
  7. 关闭连接:发送完成后,记得调用quit()方法断开与服务器的连接,释放资源。

这是一个简单的代码示例,展示了如何发送一封纯文本邮件:

import smtplib
from email.mime.text import MIMEText
from email.mime.multipart import MIMEMultipart
from email.header import Header

# --- 邮件配置信息 ---
sender_email = 'your_email@example.com'  # 发件人邮箱
sender_password = 'your_app_password'    # 发件人邮箱的授权码或应用专用密码
receiver_email = 'recipient@example.com' # 收件人邮箱
smtp_server = 'smtp.example.com'         # SMTP服务器地址 (例如:smtp.qq.com, smtp.163.com, smtp.gmail.com)
smtp_port = 465                          # SMTP服务器端口 (SSL通常是465,TLS通常是587)

# --- 构建邮件内容 ---
msg = MIMEMultipart()
msg['From'] = Header("发件人昵称", 'utf-8')
msg['To'] = Header("收件人昵称", 'utf-8')
msg['Subject'] = Header("Python邮件测试", 'utf-8')

# 邮件正文
mail_content = """
你好!

这是一封通过Python smtplib模块发送的测试邮件。
希望你能收到!

此致
敬礼
你的Python脚本
"""
msg.attach(MIMEText(mail_content, 'plain', 'utf-8'))

try:
    # --- 建立SMTP连接并发送邮件 ---
    # 对于SSL连接,使用SMTP_SSL
    server = smtplib.SMTP_SSL(smtp_server, smtp_port)
    # 如果是SMTP并需要starttls,则:
    # server = smtplib.SMTP(smtp_server, smtp_port)
    # server.ehlo()
    # server.starttls()
    # server.ehlo()

    server.login(sender_email, sender_password)
    server.send_message(msg) # 或者 server.sendmail(sender_email, receiver_email, msg.as_string())
    print("邮件发送成功!")

except smtplib.SMTPException as e:
    print(f"邮件发送失败: {e}")
except Exception as e:
    print(f"发生未知错误: {e}")
finally:
    if 'server' in locals() and server:
        server.quit()

为什么我的邮件总是发送失败?常见问题与排查

我刚开始接触Python邮件发送时,最头疼的就是邮件总是发不出去,报错信息也看得云里雾里。后来才发现,这其中有不少“坑”,但大部分都有迹可循。

一个最常见的问题就是授权码(或应用专用密码)的使用。很多邮箱服务,比如QQ邮箱、163邮箱、Gmail,为了账户安全,不允许第三方应用直接使用你的登录密码进行SMTP认证。你需要登录邮箱的网页版设置,找到“POP3/IMAP/SMTP服务”或者“客户端授权码”相关的选项,生成一个专门用于第三方客户端的授权码。这个授权码就是你在Python脚本里server.login()时要用的“密码”,而不是你平时登录邮箱的密码。这真是个让人头疼但又不得不面对的现实。

其次,SMTP服务器地址和端口配置错误也是常犯的错误。不同的邮箱服务商有不同的SMTP服务器地址(比如QQ邮箱是smtp.qq.com,163邮箱是smtp.163.com,Gmail是smtp.gmail.com)。端口也分两种主流:SSL加密通常用465,TLS加密通常用587。如果你选错了端口或者加密方式,连接自然会失败。有时候,你的网络环境或者防火墙也可能阻碍你连接到这些端口。

还有,发件人邮箱或收件人邮箱地址拼写错误,或者邮件内容被邮件服务商判定为垃圾邮件。后者尤其需要注意,如果你的邮件主题或内容过于简单、包含敏感词汇、或者发送频率过高,很可能直接被服务器拒绝或者被收件箱直接扔进垃圾箱。排查这类问题,可以先尝试发送一封最简单的纯文本邮件给自己,如果成功了,再逐步增加复杂性。

如何发送带附件、HTML内容的邮件?MIME的妙用

发送带附件或者HTML内容的邮件,就得请出email模块的MIME(Multipurpose Internet Mail Extensions)家族了。MIME这东西,听起来很技术,但其实就是个“打包盒”,它规定了邮件内容的各种类型和编码方式,让邮件客户端能够正确地解析和显示邮件。

要发送HTML内容的邮件,你只需要把MIMEText的第二个参数从'plain'改成'html'就行了。比如:MIMEText(html_content, 'html', 'utf-8')

而附件的处理则稍微复杂一些,但原理都是把附件文件读取成二进制数据,然后进行Base64编码,再封装成一个MIMEBase对象,最后添加到MIMEMultipart这个“主容器”里。你需要指定附件的文件名、MIME类型(比如图片是image/jpeg,PDF是application/pdf),以及一个Content-Disposition头,告诉邮件客户端这是一个附件。

import os
from email.mime.base import MIMEBase
from email import encoders

# ... (前面的邮件配置和MIMEMultipart初始化代码不变) ...

# 添加HTML内容
html_content = """
<html>
<body>
    <p>你好!</p>
    <p>这是一封<b>HTML格式</b>的邮件,来自Python脚本。</p>
    <p>希望你喜欢!</p>
    <img src="cid:image1.png" alt="Python Logo">
</body>
</html>
"""
msg.attach(MIMEText(html_content, 'html', 'utf-8'))

# 添加附件 (以一个图片文件为例)
# 假设当前目录下有一个 'example.png' 文件
file_path = 'example.png'
if os.path.exists(file_path):
    part = MIMEBase('application', 'octet-stream') # 通用二进制类型
    with open(file_path, 'rb') as file:
        part.set_payload(file.read())
    encoders.encode_base64(part) # 进行Base64编码

    # 设置附件的文件名,确保中文文件名显示正常
    encoded_filename = Header(os.path.basename(file_path), 'utf-8').encode()
    part.add_header('Content-Disposition', 'attachment', filename=encoded_filename)
    msg.attach(part)
else:
    print(f"警告:附件文件 '{file_path}' 不存在,将不发送附件。")

# 如果HTML中包含内联图片(cid:image1.png),需要单独添加
# from email.mime.image import MIMEImage
# with open('image1.png', 'rb') as img_file:
#     msg_image = MIMEImage(img_file.read())
#     msg_image.add_header('Content-ID', '<image1.png>') # 这里的ID要和HTML中的src="cid:..."对应
#     msg.attach(msg_image)

# ... (后面的SMTP连接和发送代码不变) ...

通过MIMEMultipart,你可以把文本、HTML、各种附件像搭积木一样组合起来,非常灵活。

大批量发送邮件需要注意什么?安全与效率考量

如果你需要通过Python发送大量的邮件,比如产品通知、营销邮件或者系统警报,那么使用个人邮箱的SMTP服务直接发送,很快就会遇到瓶颈,甚至可能导致你的邮箱被服务商暂时封禁。我之前有个项目需要发大量通知,一开始想着用自己的邮箱硬扛,结果没发几封就被封了。后来才意识到,专业的事还得专业工具来办。

首先,个人邮箱的SMTP服务通常有严格的发送频率和数量限制。你不可能指望通过一个个人邮箱一天发送几万封邮件。一旦触发了这些限制,你的邮件可能会被退回,甚至你的账户可能会被标记为垃圾邮件发送者。

其次,邮件送达率是个大问题。自己发送的邮件,很可能被收件方的垃圾邮件过滤器拦截,最终根本到不了用户的收件箱。专业的邮件服务提供商(ESP,Email Service Provider),比如SendGrid、Mailgun、或者国内的腾讯邮件推送、阿里云邮件推送等,它们有专门的IP信誉管理、反垃圾邮件机制和投递优化技术,能大大提高邮件的送达率。虽然它们通常是付费服务,但对于商业或大规模应用来说,这是非常值得的投资。

如果你确实需要自己处理,并且量不大,那么有几个点可以考虑:

  • 控制发送频率:在发送每封邮件之间加入适当的time.sleep(),避免短时间内发送大量邮件。
  • 异常处理和重试机制:网络波动、服务器临时故障都可能导致发送失败。为你的发送逻辑加上健壮的try-except块,并在失败时考虑重试几次。
  • 日志记录:记录每封邮件的发送状态,包括成功、失败原因等,方便后续排查问题。
  • 安全存储凭证:不要将你的邮箱授权码直接写死在代码里,尤其是在公共代码库中。考虑使用环境变量、配置文件或更安全的密钥管理服务来存储这些敏感信息。

总之,对于小规模的通知或个人使用,smtplib已经足够强大。但一旦涉及到“大批量”这个词,就得认真考虑专业的邮件服务了,它们不仅能解决发送效率和送达率的问题,还能提供详细的发送报告和统计分析,这些都是自己搭建无法比拟的优势。

以上就是《Python发送邮件教程:smtplib使用全解析》的详细内容,更多关于Python,邮件发送,授权码,smtplib,email模块的资料请关注golang学习网公众号!

CSS直接子元素选择器详解CSS直接子元素选择器详解
上一篇
CSS直接子元素选择器详解
Selenium延迟启动Chrome优化加载体验
下一篇
Selenium延迟启动Chrome优化加载体验
查看更多
最新文章
查看更多
课程推荐
  • 前端进阶之JavaScript设计模式
    前端进阶之JavaScript设计模式
    设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
    542次学习
  • GO语言核心编程课程
    GO语言核心编程课程
    本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
    511次学习
  • 简单聊聊mysql8与网络通信
    简单聊聊mysql8与网络通信
    如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
    498次学习
  • JavaScript正则表达式基础与实战
    JavaScript正则表达式基础与实战
    在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
    487次学习
  • 从零制作响应式网站—Grid布局
    从零制作响应式网站—Grid布局
    本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
    484次学习
查看更多
AI推荐
  • 千音漫语:智能声音创作助手,AI配音、音视频翻译一站搞定!
    千音漫语
    千音漫语,北京熠声科技倾力打造的智能声音创作助手,提供AI配音、音视频翻译、语音识别、声音克隆等强大功能,助力有声书制作、视频创作、教育培训等领域,官网:https://qianyin123.com
    117次使用
  • MiniWork:智能高效AI工具平台,一站式工作学习效率解决方案
    MiniWork
    MiniWork是一款智能高效的AI工具平台,专为提升工作与学习效率而设计。整合文本处理、图像生成、营销策划及运营管理等多元AI工具,提供精准智能解决方案,让复杂工作简单高效。
    112次使用
  • NoCode (nocode.cn):零代码构建应用、网站、管理系统,降低开发门槛
    NoCode
    NoCode (nocode.cn)是领先的无代码开发平台,通过拖放、AI对话等简单操作,助您快速创建各类应用、网站与管理系统。无需编程知识,轻松实现个人生活、商业经营、企业管理多场景需求,大幅降低开发门槛,高效低成本。
    128次使用
  • 达医智影:阿里巴巴达摩院医疗AI影像早筛平台,CT一扫多筛癌症急慢病
    达医智影
    达医智影,阿里巴巴达摩院医疗AI创新力作。全球率先利用平扫CT实现“一扫多筛”,仅一次CT扫描即可高效识别多种癌症、急症及慢病,为疾病早期发现提供智能、精准的AI影像早筛解决方案。
    121次使用
  • 智慧芽Eureka:更懂技术创新的AI Agent平台,助力研发效率飞跃
    智慧芽Eureka
    智慧芽Eureka,专为技术创新打造的AI Agent平台。深度理解专利、研发、生物医药、材料、科创等复杂场景,通过专家级AI Agent精准执行任务,智能化工作流解放70%生产力,让您专注核心创新。
    126次使用
微信登录更方便
  • 密码登录
  • 注册账号
登录即同意 用户协议隐私政策
返回登录
  • 重置密码