Python使用POP3和SMTP协议收发邮件的示例代码


Posted in Python onApril 16, 2019

先来了解一下收/发邮件有哪些协议:

SMTP协议
SMTP(Simple Mail Transfer Protocol),即简单邮件传输协议。相当于中转站,将邮件发送到客户端。

POP3协议
POP3(Post Office Protocol 3),即邮局协议的第3个版本,是电子邮件的第一个离线协议标准。该协议把邮件下载到本地计算机,不与服务器同步,缺点是更易丢失邮件或多次下载相同的邮件。

IMAP协议
IMAP(Internet Mail Access Protocol),即交互式邮件存取协议。该协议连接远程邮箱直接操作,与服务器内容同步。

Exchange服务
Exchange服务是一个设计完备的邮件服务器产品,提供了通常所需要的全部邮件服务功能。除了常规SMTP/POP协议服务之外,它还支持IMAP4 、LDAP和NNTP协议。

Python内置对SMTP/POP3/IMAP的支持。更多详情请移步Python官方教程

SMTP发送邮件

Python对SMTP支持有 smtplibemail 两个模块, email 负责构造邮件, smtplib 负责发送邮件。

构造邮件

构造最简单的纯文本邮件,如下:

from email.mime.text import MIMEText

msg = MIMEText('hello, send by Python...', 'plain', 'utf-8')

注意到构造 MIMEText 对象时,第一个参数就是邮件正文,第二个参数是MIME的subtype,传入 'plain' 表示纯文本,最终的MIME就是 'text/plain' ,最后一定要用 utf-8 编码保证多语言兼容性。

发送邮件

import smtplib

# 输入Email地址和口令:
from_addr = 'test_from_addr@qq.com'
password = 'Password'
# 输入收件人地址:
to_addr = 'test_to_addr@qq.com'
# 输入SMTP服务器地址:
smtp_server = smtp.qq.com

server = smtplib.SMTP(smtp_server, 25) # SMTP协议默认端口是25
# server.starttls() # 如果是SSL,则用 587 端口,再加上这句代码就行了
server.set_debuglevel(1) # 打印出和SMTP服务器交互的所有信息
server.login(from_addr, password) # 登录SMTP服务器
server.sendmail(from_addr, [to_addr], msg.as_string()) # 发邮件
server.quit()

sendmail() 方法就是发邮件,由于可以一次发给多个人,所以传入一个 list ,邮件正文是一个 stras_string() 把MIMEText对象变成 str

注意:QQ邮件等需要手动开通 SMTP服务 , 邮箱设置 => 账号 => POP3/SMTP服务,如下图:

Python使用POP3和SMTP协议收发邮件的示例代码
Python使用POP3和SMTP协议收发邮件的示例代码

此时,我们就可以收到邮件了,如下:

Python使用POP3和SMTP协议收发邮件的示例代码 

添加邮件标题、收/发件人

邮件主题、显示发件人、收件人等信息并不是通过SMTP协议发送的,而是包含在 MIMEText 对象中,如下:

from email import encoders
from email.header import Header
from email.mime.text import MIMEText
from email.utils import parseaddr, formataddr

import smtplib

def _format_addr(s):
 name, addr = parseaddr(s)
 return formataddr((Header(name, 'utf-8').encode(), addr))

from_addr = 'test_from_addr@qq.com'
password = 'Password'
to_addr = 'test_to_addr@qq.com'
smtp_server = smtp.qq.com

msg = MIMEText('hello, send by Python...', 'plain', 'utf-8')
msg['From'] = _format_addr('发件人昵称 <%s>' % from_addr)
msg['To'] = _format_addr('收件人昵称 <%s>' % to_addr)
msg['Subject'] = Header('这是个有主题的邮件', 'utf-8').encode()

server = smtplib.SMTP(smtp_server, 25)
server.set_debuglevel(1)
server.login(from_addr, password)
server.sendmail(from_addr, [to_addr], msg.as_string())
server.quit()

收到的邮件,如下:

Python使用POP3和SMTP协议收发邮件的示例代码

收件人并不是我们设置的 “收件人昵称”,是因为很多邮件服务商在显示邮件时,会把收件人名字自动替换为用户注册的名字,这无伤大雅。

发送HTML邮件

要发送HTML邮件很简单,在构造 MIMEText 对象时,把HTML字符串传进去,再把第二个参数由 plain 变为 html ,如下:

msg = MIMEText('<html><body><h1>Hello</h1>' +
 '<p>send by <a href="http://blog.pangao.vip">PanGao's blog</a>...</p>' +
 '</body></html>', 'html', 'utf-8')

发送附件

要想发送附件,需要构造一个 MIMEMultipart 对象代表邮件本身,然后往里面加上一个 MIMEText 作为邮件正文,再继续往里面加上表示附件的 MIMEBase 对象,如下:

from email.mime.multipart import MIMEMultipart
from email.mime.base import MIMEBase

# 邮件对象:
msg = MIMEMultipart()
msg['From'] = _format_addr('发件人昵称 <%s>' % from_addr)
msg['To'] = _format_addr('收件人昵称 <%s>' % to_addr)
msg['Subject'] = Header('这是个有主题的邮件', 'utf-8').encode()

# 邮件正文是MIMEText:
msg.attach(MIMEText('send with file...', 'plain', 'utf-8'))

# 添加附件就是加上一个MIMEBase,从本地读取一个图片:
with open('/Users/pangao/Downloads/test.png', 'rb') as f:
 # 设置附件的MIME和文件名,这里是png类型:
 mime = MIMEBase('image', 'png', filename='test.png')
 # 加上必要的头信息:
 mime.add_header('Content-Disposition', 'attachment', filename='test.png')
 mime.add_header('Content-ID', '<0>')
 mime.add_header('X-Attachment-Id', '0')
 # 把附件的内容读进来:
 mime.set_payload(f.read())
 # 用Base64编码:
 encoders.encode_base64(mime)
 # 添加到MIMEMultipart:
 msg.attach(mime)

Python使用POP3和SMTP协议收发邮件的示例代码

发送图片

由于 mac 自带的邮件会自动把图片附件插入邮件正文中,所以样式很好看。但是普通邮件可能没这么便捷(抱歉,我没见过普通邮件。。。小小得瑟一下)

如果要把一个图片嵌入到邮件正文中怎么做?直接在HTML邮件中链接图片地址行不行?答案是,大部分邮件服务商都会自动屏蔽带有外链的图片,因为不知道这些链接是否指向恶意网站。

要把图片嵌入到邮件正文中,我们只需按照发送附件的方式,先把邮件作为附件添加进去,然后,在HTML中通过引用 src="cid:0" 就可以把附件作为图片嵌入了。如果有多个图片,给它们依次编号,然后引用不同的 cid:x 即可。

把上面代码加入 MIMEMultipartMIMETextplain 改为 html ,然后在适当的位置引用图片,如下:

msg.attach(MIMEText('<html><body><h1>Hello</h1>' +
 '<p><img src="cid:0"></p>' +
 '</body></html>', 'html', 'utf-8'))

同时支持HTML和Plain格式

如果我们发送HTML邮件,收件人通过浏览器或者Outlook之类的软件是可以正常浏览邮件内容的,但是,如果收件人使用的设备太古老,查看不了HTML邮件怎么办?

办法是在发送HTML的同时再附加一个纯文本,如果收件人无法查看HTML格式的邮件,就可以自动降级查看纯文本邮件。

利用 MIMEMultipart 就可以组合一个HTML和Plain,要注意指定subtype是 alternative ,如下:

msg = MIMEMultipart('alternative')
msg['From'] = ...
msg['To'] = ...
msg['Subject'] = ...

msg.attach(MIMEText('hello', 'plain', 'utf-8'))
msg.attach(MIMEText('<html><body><h1>Hello</h1></body></html>', 'html', 'utf-8'))
# 正常发送msg对象...

加密SMTP

使用标准的25端口连接SMTP服务器时,使用的是明文传输,发送邮件的整个过程可能会被窃听。要更安全地发送邮件,可以加密SMTP会话,实际上就是先创建SSL安全连接,然后再使用SMTP协议发送邮件。

某些邮件服务商,例如Gmail,提供的SMTP服务必须要加密传输。我们来看看如何通过Gmail提供的安全SMTP发送邮件。

必须知道,Gmail的SMTP端口是587,因此,修改代码如下:

smtp_server = 'smtp.gmail.com'
smtp_port = 587
server = smtplib.SMTP(smtp_server, smtp_port)
server.starttls()
# 剩下的代码和前面的一模一样:
server.set_debuglevel(1)
...

只需要在创建 SMTP 对象后,立刻调用 starttls() 方法,就创建了安全连接。后面的代码和前面的发送邮件代码完全一样。

POP3收取邮件

Python内置一个 poplib 模块,实现了POP3协议,可以直接用来收邮件。

注意到POP3协议收取的不是一个已经可以阅读的邮件本身,而是邮件的原始文本,这和SMTP协议很像,SMTP发送的也是经过编码后的一大段文本。

要把POP3收取的文本变成可以阅读的邮件,还需要用email模块提供的各种类来解析原始文本,变成可阅读的邮件对象。

所以,收取邮件分两步:

第一步:用 poplib 把邮件的原始文本下载到本地;

第二部:用 email 解析原始文本,还原为邮件对象。

通过POP3下载邮件

POP3协议本身很简单,以下面的代码为例,我们来获取最新的一封邮件内容:

from email.parser import Parser
import poplib

# 输入邮件地址, 口令和POP3服务器地址:
email = 'pangao1990@qq.com'
password = 'Password'
pop3_server = 'pop.qq.com'

# 连接到POP3服务器:
server = poplib.POP3_SSL(pop3_server)
# 可以打开或关闭调试信息:
server.set_debuglevel(1)

# 身份认证:
server.user(email)
server.pass_(password)

# list()返回所有邮件的编号:
resp, mails, octets = server.list()

# 获取最新一封邮件, 注意索引号从1开始:
index = len(mails)
resp, lines, octets = server.retr(index)

# lines存储了邮件的原始文本的每一行,
# 可以获得整个邮件的原始文本:
msg_content = b'\r\n'.join(lines).decode('utf-8')
# 稍后解析出邮件:
msg = Parser().parsestr(msg_content)

# 可以根据邮件索引号直接从服务器删除邮件:
# server.dele(index)
# 关闭连接:
server.quit()

但是这个 Message 对象本身可能是一个 MIMEMultipart 对象,即包含嵌套的其他 MIMEBase 对象,嵌套可能还不止一层。

所以我们要递归地打印出 Message 对象的层次结构:

from email.header import decode_header
from email.utils import parseaddr


def print_info(msg, indent=0):
 if indent == 0:
  for header in ['From', 'To', 'Subject']:
   value = msg.get(header, '')
   if value:
    if header == 'Subject':
     value = decode_str(value)
    else:
     hdr, addr = parseaddr(value)
     name = decode_str(hdr)
     value = u'%s <%s>' % (name, addr)
   print('%s%s: %s' % (' ' * indent, header, value))
 if (msg.is_multipart()):
  parts = msg.get_payload()
  for n, part in enumerate(parts):
   print('%spart %s' % (' ' * indent, n))
   print('%s--------------------' % (' ' * indent))
   print_info(part, indent + 1)
 else:
  content_type = msg.get_content_type()
  if content_type == 'text/plain' or content_type == 'text/html':
   content = msg.get_payload(decode=True)
   charset = guess_charset(msg)
   if charset:
    content = content.decode(charset)
   print('%sText: %s' % (' ' * indent, content + '...'))
  else:
   print('%sAttachment: %s' % (' ' * indent, content_type))


def decode_str(s):
 value, charset = decode_header(s)[0]
 if charset:
  value = value.decode(charset)
 return value


def guess_charset(msg):
 charset = msg.get_charset()
 if charset is None:
  content_type = msg.get('Content-Type', '').lower()
  pos = content_type.find('charset=')
  if pos >= 0:
   charset = content_type[pos + 8:].strip()
 return charset


print_info(msg) #解析

# From: 木叶 <pangao1990@qq.com>
# To: <mail@pangao.vip>
# Subject: 测试主题
# Text: 测试内容
# 
# ...

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Python 相关文章推荐
Python 不同对象比较大小示例探讨
Aug 21 Python
python开发环境PyScripter中文乱码问题解决方案
Sep 11 Python
python模仿网页版微信发送消息功能
Feb 24 Python
python实现AES加密与解密
Mar 28 Python
Python配置虚拟环境图文步骤
May 20 Python
Django 请求Request的具体使用方法
Nov 11 Python
python支持多线程的爬虫实例
Dec 21 Python
通过实例学习Python Excel操作
Jan 06 Python
Django 删除upload_to文件的步骤
Mar 30 Python
Python中的全局变量如何理解
Jun 04 Python
python读取图片颜色值并生成excel像素画的方法实例
Feb 19 Python
Python爬虫网络请求之代理服务器和动态Cookies
Apr 12 Python
Python这样操作能存储100多万行的xlsx文件
Apr 16 #Python
Python字符串内置函数功能与用法总结
Apr 16 #Python
python3实现字符串操作的实例代码
Apr 16 #Python
几个适合python初学者的简单小程序,看完受益匪浅!(推荐)
Apr 16 #Python
Django之无名分组和有名分组的实现
Apr 16 #Python
Pythony运维入门之Socket网络编程详解
Apr 15 #Python
使用python实现抓取腾讯视频所有电影的爬虫
Apr 15 #Python
You might like
利用discuz实现PHP大文件上传应用实例代码
2008/11/14 PHP
PHP投票系统防刷票判断流程分析
2012/02/04 PHP
通过PHP的内置函数,通过DES算法对数据加密和解密
2012/06/21 PHP
获取php页面执行时间,数据库读写次数,函数调用次数等(THINKphp)
2013/06/03 PHP
PHP is_subclass_of函数的一个BUG和解决方法
2014/06/01 PHP
PHP网页游戏学习之Xnova(ogame)源码解读(二)
2014/06/23 PHP
PHP根据两点间的经纬度计算距离
2014/10/31 PHP
phpstorm配置Xdebug进行调试PHP教程
2014/12/01 PHP
PHP实现随机数字、字母的验证码功能
2018/08/01 PHP
js日期相关函数总结分享
2013/10/15 Javascript
jQuery判断元素上是否绑定了指定事件的方法
2015/03/17 Javascript
jquery UI Datepicker时间控件的使用方法(终结版)
2015/11/07 Javascript
AngularJs验证重复密码的方法(两种)
2016/11/25 Javascript
浅谈vue中数据双向绑定的实现原理
2017/09/14 Javascript
基于Koa(nodejs框架)对json文件进行增删改查的示例代码
2019/02/02 NodeJs
VUE实现密码验证与提示功能
2019/10/18 Javascript
uni-app如何实现增量更新功能
2020/01/03 Javascript
[01:00:30]TFT vs VGJ.T Supermajor 败者组 BO3 第一场 6.5
2018/06/06 DOTA
Python中类的继承代码实例
2014/10/28 Python
Python基于二分查找实现求整数平方根的方法
2016/05/12 Python
详解Python中表达式i += x与i = i + x是否等价
2017/02/08 Python
python匿名函数用法实例分析
2019/08/03 Python
Python字符串处理的8招秘籍(小结)
2019/08/13 Python
Python中注释(多行注释和单行注释)的用法实例
2019/08/28 Python
kafka-python 获取topic lag值方式
2019/12/23 Python
python Xpath语法的使用
2020/11/26 Python
波兰数码相机及配件网上商店: Cyfrowe.pl
2017/06/19 全球购物
一个J2EE项目团队的主要人员组成是什么
2012/06/04 面试题
2014年社区重阳节活动策划方案
2014/09/16 职场文书
2014年企业党支部工作总结
2014/12/04 职场文书
学年个人总结范文
2015/03/05 职场文书
结婚当天新郎保证书
2015/05/08 职场文书
MongoDB使用profile分析慢查询的步骤
2021/04/30 MongoDB
Pandas实现批量拆分与合并Excel的示例代码
2022/05/30 Python
类和原型的设计模式之复制与委托差异
2022/07/07 Javascript
windows系统搭建WEB服务器详细教程
2022/08/05 Servers