Python利用SMTP发送邮件的常见问题与解决方案详解

 更新时间:2025年04月21日 08:12:24   作者:码农阿豪@新空间  
在自动化办公和系统开发中,邮件发送功能是常见的需求,本文将从实际案例出发,分析常见的SMTP错误,并提供完整的解决方案,希望对大家有一定的帮助

引言

在自动化办公和系统开发中,邮件发送功能是常见的需求。无论是发送通知、报告,还是传输文件,Python的smtplib和email库提供了便捷的实现方式。然而,在实际开发中,开发者常会遇到各种SMTP错误,如(-1, b'\x00\x00\x00')、AttributeError('characters_written')等,这些问题往往让人束手无策。

本文将从实际案例出发,分析常见的SMTP错误,并提供完整的解决方案,帮助开发者快速定位和解决问题。同时,我们还会优化代码,使其更健壮、更易于维护。

1. 问题背景

在Python中,使用smtplib发送邮件时,通常会遇到两类问题:

  • 连接与认证问题(如授权码错误、SMTP服务器拒绝连接)
  • 数据编码与协议问题(如SSL/TLS握手失败、附件编码错误)

以下是一个典型的邮件发送代码,它可能会触发上述错误:

import smtplib
from email.mime.text import MIMEText
from email.mime.multipart import MIMEMultipart
from email.mime.application import MIMEApplication
import os
import logging

logger = logging.getLogger(__name__)

def send_email_with_attachment(filepath, receiver_email):
    sender_email = "your_email@qq.com"
    password = "your_authorization_code"  # QQ邮箱授权码
    smtp_server = "smtp.qq.com"
    smtp_port = 465  # QQ邮箱SSL端口

    msg = MIMEMultipart()
    msg['From'] = sender_email
    msg['To'] = receiver_email
    msg['Subject'] = "文件处理结果通知"

    # 邮件正文
    msg.attach(MIMEText("请查收附件", 'plain'))

    # 添加附件
    try:
        with open(filepath, "rb") as f:
            part = MIMEApplication(f.read(), Name=os.path.basename(filepath))
            part['Content-Disposition'] = f'attachment; filename="{os.path.basename(filepath)}"'
            msg.attach(part)
    except FileNotFoundError:
        logger.error("附件文件不存在")
        return False

    # 发送邮件
    try:
        with smtplib.SMTP_SSL(smtp_server, smtp_port) as server:
            server.login(sender_email, password)
            server.sendmail(sender_email, receiver_email, msg.as_string())
        return True
    except Exception as e:
        logger.error(f"发送邮件失败: {e}", exc_info=True)
        return False

运行此代码时,可能会遇到以下错误:

2. 常见错误及解决方案

错误1:SMTPResponseException: (-1, b'\x00\x00\x00')

错误原因

  • 授权码错误或已过期。
  • SMTP服务器(如QQ邮箱)检测到异常登录,拒绝连接。
  • 网络或防火墙拦截了SMTP请求。

解决方案

1.检查授权码

登录QQ邮箱 → 设置 → 账户 → 生成新的授权码,并替换代码中的password。

2.更换SMTP端口(推荐使用587+TLS)

QQ邮箱支持465(SSL)和587(TLS),后者更稳定:

smtp_port = 587  # 改用TLS端口
with smtplib.SMTP(smtp_server, smtp_port) as server:
    server.starttls()  # 启用TLS加密
    server.login(sender_email, password)

3.检查网络环境

关闭防火墙或切换网络(如使用手机热点测试)。

错误2:AttributeError('characters_written')

错误原因

  • Python 3.10+ 与 SMTP_SSL 的兼容性问题。
  • SSL/TLS握手失败,可能是由于OpenSSL版本不匹配。

解决方案

1.改用starttls()(推荐)

避免使用SMTP_SSL,改用SMTP + starttls():

with smtplib.SMTP(smtp_server, 587) as server:
    server.starttls()  # 显式启用TLS
    server.login(sender_email, password)

2.降级Python或升级依赖库

如果仍报错,尝试降级到Python 3.9,或更新pyopenssl:

pip install --upgrade pyopenssl

3. 优化后的邮件发送代码

结合上述解决方案,优化后的代码如下:

import smtplib
from email.mime.text import MIMEText
from email.mime.multipart import MIMEMultipart
from email.mime.application import MIMEApplication
import os
import logging
import base64

logger = logging.getLogger(__name__)

def send_email_with_attachment(filepath, receiver_email):
    """发送带附件的邮件(支持QQ邮箱)"""
    sender_email = "your_email@qq.com"
    password = "your_authorization_code"  # 替换为最新授权码
    smtp_server = "smtp.qq.com"
    smtp_port = 587  # 使用TLS端口

    # 检查附件是否存在
    if not os.path.exists(filepath):
        logger.error(f"附件文件不存在: {filepath}")
        return False

    # 创建邮件内容
    msg = MIMEMultipart()
    msg['From'] = sender_email
    msg['To'] = receiver_email
    msg['Subject'] = "文件处理结果通知"
    msg.attach(MIMEText("请查收附件", 'plain'))

    # 添加附件
    try:
        with open(filepath, "rb") as f:
            part = MIMEApplication(f.read(), Name=os.path.basename(filepath))
            part['Content-Disposition'] = f'attachment; filename="{os.path.basename(filepath)}"'
            msg.attach(part)
    except Exception as e:
        logger.error(f"附件处理失败: {e}")
        return False

    # 发送邮件(使用TLS)
    try:
        with smtplib.SMTP(smtp_server, smtp_port) as server:
            server.starttls()  # 启用TLS加密
            server.login(sender_email, password)
            server.sendmail(sender_email, receiver_email, msg.as_string())
        logger.info("邮件发送成功")
        return True
    except smtplib.SMTPException as e:
        logger.error(f"SMTP错误: {e}")
    except Exception as e:
        logger.error(f"未知错误: {e}", exc_info=True)
    return False

优化点

1.更健壮的异常处理

区分SMTPException和其他异常,便于排查问题。

2.附件预检查

发送前验证附件是否存在。

3.日志记录

使用logging记录详细错误信息。

4. SMTP调试技巧

手动测试SMTP连接

使用openssl命令行工具测试SMTP服务是否可用:

openssl s_client -connect smtp.qq.com:587 -starttls smtp -crlf

输入以下命令(替换为你的邮箱和授权码):

EHLO test
AUTH LOGIN
<base64编码的邮箱>  # 示例:echo -n "your_email@qq.com" | base64
<base64编码的授权码>
QUIT

如果返回235 Authentication successful,说明SMTP配置正确。

检查防火墙

在Windows上,运行以下命令放行SMTP端口:

netsh advfirewall firewall add rule name="SMTP" dir=in action=allow protocol=TCP localport=587

5. 总结与最佳实践

常见问题总结

错误原因解决方案
(-1, b'\x00\x00\x00')授权码错误/SMTP拒绝更新授权码,改用starttls()
AttributeError('characters_written')Python 3.10+兼容性问题降级Python或改用SMTP + starttls()
SSL: WRONG_VERSION_NUMBERSSL/TLS配置错误使用starttls() + 端口587

最佳实践

  • 使用TLS(端口587)代替SSL(端口465),兼容性更好。
  • 定期更新授权码,避免因过期导致发送失败。
  • 添加日志记录,便于排查问题。
  • 手动测试SMTP连接,确保服务器可用。

通过本文的分析和优化,你应该能够解决大多数Python邮件发送问题。如果你的场景涉及更复杂的需求(如批量发送、HTML邮件),可以进一步扩展代码逻辑。 

到此这篇关于Python利用SMTP发送邮件的常见问题与解决方案详解的文章就介绍到这了,更多相关Python SMTP发送邮件内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 在python中使用xlrd获取合并单元格的方法

    在python中使用xlrd获取合并单元格的方法

    今天小编就为大家分享一篇在python中使用xlrd获取合并单元格的方法,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2018-12-12
  • 解决python matplotlib imshow无法显示的问题

    解决python matplotlib imshow无法显示的问题

    今天小编就为大家分享一篇解决python matplotlib imshow无法显示的问题,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2018-05-05
  • python下10个简单实例代码

    python下10个简单实例代码

    最近学python比较顺手,找到感觉了,所以,我想把我用来练习的实例题目分享出来,有兴趣的朋友可以关注一下。 文章分为10篇,每篇10题,共100道实例。后续如果需要可以增加
    2017-11-11
  • Python通过命令行向Scrapy传递参数

    Python通过命令行向Scrapy传递参数

    crapy作为一个强大的Web爬取框架,提供了灵活的命令行参数传递功能,本文介绍了通过命令行向Scrapy爬虫传递参数的方法,旨在增强爬虫的灵活性和可配置性,感兴趣的可以了解一下
    2024-10-10
  • Python编程密码学文件加密与解密代码解析

    Python编程密码学文件加密与解密代码解析

    这篇文章主要为大家介绍了Python编程密码学文件加密与解密,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-05-05
  • 一文分享Python自动化巡检脚本

    一文分享Python自动化巡检脚本

    文章详细介绍了自动化巡检脚本的开发过程,从设备清单、巡检命令、并线多线程并发巡检到生成Excel报告,并推荐了Netmiko、NAPALM、ntc-templates、TextFSM、Paramiko等五个网络自动化工具,可以帮助工程师提高工作效率
    2026-05-05
  • Python使用type动态创建类操作示例

    Python使用type动态创建类操作示例

    这篇文章主要介绍了Python使用type动态创建类操作,结合实例形式详细分析了Python使用type动态创建类的具体原理、实现方法与操作注意事项,需要的朋友可以参考下
    2020-02-02
  • python redis 删除key脚本的实例

    python redis 删除key脚本的实例

    今天小编就为大家分享一篇python redis 删除key脚本的实例,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2019-02-02
  • Python机器学习实战之k-近邻算法的实现

    Python机器学习实战之k-近邻算法的实现

    k-近邻算法采用测量不同特征值之间的距离方法进行分类。这篇文章主要为大家介绍了如何通过python实现K近邻算法,有需要的朋友可以借鉴参考下,希望能够有所帮助
    2021-11-11
  • Python Flask RESTful使用demo演示

    Python Flask RESTful使用demo演示

    这篇文章主要为大家介绍了Python Flask RESTful使用demo演示,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-03-03

最新评论