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 相关文章推荐
使用graphics.py实现2048小游戏
Mar 10 Python
Python常用的内置序列结构(列表、元组、字典)学习笔记
Jul 08 Python
Python urls.py的三种配置写法实例详解
Apr 28 Python
详谈Python高阶函数与函数装饰器(推荐)
Sep 30 Python
使用python脚本实现查询火车票工具
Jul 19 Python
python画柱状图--不同颜色并显示数值的方法
Dec 13 Python
selenium+python自动化测试之鼠标和键盘事件
Jan 23 Python
Python datetime 格式化 明天,昨天实例
Mar 02 Python
python3 xpath和requests应用详解
Mar 06 Python
学习Python爬虫的几点建议
Aug 05 Python
python自动生成证件号的方法示例
Jan 14 Python
PyMongo 查询数据的实现
Jun 28 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
深入解析php模板技术原理【一】
2008/01/10 PHP
php数组删除元素示例
2014/03/21 PHP
PHP文件生成的图片无法使用CDN缓存的解决方法
2015/06/20 PHP
CI框架中数据库操作函数$this-&gt;db-&gt;where()相关用法总结
2016/05/17 PHP
PHP实现实时生成并下载超大数据量的EXCEL文件详解
2017/10/23 PHP
php app支付宝回调(异步通知)详解
2018/07/25 PHP
PHP进阶学习之命名空间基本用法分析
2019/06/18 PHP
轻轻松松学习JavaScript
2007/02/25 Javascript
用函数式编程技术编写优美的 JavaScript_ibm
2008/05/16 Javascript
window.onbeforeunload方法在IE下无法正常工作的解决办法
2010/01/23 Javascript
Jquery 扩展方法
2010/05/06 Javascript
jQuery的context属性用法实例
2014/12/27 Javascript
js获取当前日期时间及其它操作汇总
2015/04/17 Javascript
基于BootStrap Metronic开发框架经验小结【七】数据的导入、导出及附件的查看处理
2016/05/12 Javascript
jQuery EasyUI框架中的Datagrid数据表格组件结构详解
2016/06/09 Javascript
Servlet实现文件上传,可多文件上传示例
2016/12/05 Javascript
实例解析Array和String方法
2016/12/14 Javascript
jQuery时间验证和转换为标准格式的时间格式
2017/03/06 Javascript
for循环 + setTimeout 结合一些示例(前端面试题)
2017/08/30 Javascript
python获得图片base64编码示例
2014/01/16 Python
python模拟表单提交登录图书馆
2018/04/27 Python
Python中.join()和os.path.join()两个函数的用法详解
2018/06/11 Python
Django网络框架之创建虚拟开发环境操作示例
2019/06/06 Python
html5使用canvas压缩图片的示例代码
2018/09/11 HTML / CSS
中国最大隐形眼镜网上商城:视客眼镜网
2016/10/30 全球购物
荷兰网上药店:Drogisterij.net
2019/09/03 全球购物
俄罗斯品牌服装和鞋子在线商店:BRIONITY
2020/03/26 全球购物
十岁生日同学答谢词
2014/01/19 职场文书
财务人员求职自荐书范文
2014/02/10 职场文书
南京市纪委监察局整改方案
2014/09/16 职场文书
教师个人年度总结
2015/02/11 职场文书
2015年感恩父亲节活动策划方案
2015/05/05 职场文书
电话营销开场白
2015/05/29 职场文书
2016教师廉洁教育心得体会
2016/01/13 职场文书
探究Mysql模糊查询是否区分大小写
2021/06/11 MySQL
css之clearfix的用法深入理解(必看篇)
2023/05/21 HTML / CSS