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封装对象实现时间效果
Apr 23 Python
python通过pil模块将raw图片转换成png图片的方法
Mar 16 Python
Python中的类与对象之描述符详解
Mar 27 Python
Python中的choice()方法使用详解
May 15 Python
python实现发送和获取手机短信验证码
Jan 15 Python
Python实现的递归神经网络简单示例
Aug 11 Python
详解python字节码
Feb 07 Python
Python 读取图片文件为矩阵和保存矩阵为图片的方法
Apr 27 Python
python爬虫之urllib,伪装,超时设置,异常处理的方法
Dec 19 Python
使用Python中的reduce()函数求积的实例
Jun 28 Python
python dataframe NaN处理方式
Dec 26 Python
Django分组聚合查询实例分享
Apr 29 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
无线电波是什么?它是怎样传输的?
2021/03/01 无线电
PHP实现异步调用方法研究与分享
2011/10/27 PHP
php中让上传的文件大小在上传前就受限制的两种解决方法
2013/06/24 PHP
php中Array2xml类实现数组转化成XML实例
2014/12/08 PHP
PHP中防止SQL注入方法详解
2014/12/25 PHP
浅谈COOKIE和SESSION区别
2015/07/19 PHP
记录Yii2框架开发微信公众号遇到的问题及解决方法
2018/07/20 PHP
PHP addAttribute()函数讲解
2019/02/03 PHP
js 通过cookie实现刷新不变化树形菜单
2014/10/30 Javascript
深入浅析Extjs中store分组功能的使用方法
2016/04/20 Javascript
深入理解JavaScript中的块级作用域、私有变量与模块模式
2016/10/31 Javascript
Javascript 两种刷新方法以及区别和适用范围
2017/01/17 Javascript
详解webpack的配置文件entry与output
2017/08/21 Javascript
vuejs项目打包之后的首屏加载优化及打包之后出现的问题
2018/04/01 Javascript
傻瓜式解读koa中间件处理模块koa-compose的使用
2018/10/30 Javascript
nodejs中request库使用HTTPS代理的方法
2019/04/30 NodeJs
[02:08:58]2014 DOTA2国际邀请赛中国区预选赛 Ne VS CIS
2014/05/22 DOTA
Python open读写文件实现脚本
2008/09/06 Python
Python3.x中自定义比较函数
2015/04/24 Python
如何利用Fabric自动化你的任务
2016/10/20 Python
numpy.delete删除一列或多列的方法
2018/04/03 Python
Python模拟FTP文件服务器的操作方法
2020/02/18 Python
基于Python数据结构之递归与回溯搜索
2020/02/26 Python
Python faker生成器生成虚拟数据代码实例
2020/07/20 Python
Python3.9新特性详解
2020/10/10 Python
全球知名旅游社区法国站点:TripAdvisor法国
2016/08/03 全球购物
Java里面如何把一个Array数组转换成Collection, List
2013/07/26 面试题
java程序员面试交流
2012/11/29 面试题
师范生自荐信范文
2013/10/06 职场文书
商务专员岗位职责
2013/11/23 职场文书
保密协议书范本
2014/04/22 职场文书
人力资源管理求职信
2014/08/07 职场文书
反对四风自我剖析材料
2014/10/07 职场文书
从严治党主题教育活动总结
2015/05/07 职场文书
导游词之神仙居景区
2019/11/15 职场文书
Pandas实现批量拆分与合并Excel的示例代码
2022/05/30 Python