通过python发送带有附件、图片、表格的邮件


最近的工作中经常需要监控一些任务或服务,以期望及时发现问题、或者方便看到工作的进展或成果,这时候通过自动发送邮件就是一个好的解决方案。同时,也有小伙伴求助如何发送带图片或表格或附件的邮件,那么这篇文章就做一个教程,方便忘记时做参考。

邮件是通过SMTP(简单邮件传输协议)传送的,python的smtplib模块对smtp协议进行了简单的封装,即smtplib中的SMTP类。

通过smtplib模块发送邮件主要分为四个步骤:

  1. 实例化SMTP对象
  2. 连接SMTP服务器
  3. 登录账号
  4. 编辑邮件内容
  5. 发送邮件

纯文本邮件

先看一个简单的代码例子

## 导入模块
from email.mime.text import MIMEText
from email.mime.multipart import MIMEMultipart
import smtplib

mail_host = "smtp.163.com" #SMTP服务器地址
mail_sender = "****@163.com" #账号
mail_passwd = "****" #密码

## 构造邮件内容
msg = MIMEText("这里是邮件正文内容,只是测试一下",'plain','utf-8')
msg["Subject"] = "这里是邮件主题"
msg["From"] = mail_sender #发送人
msg["To"] = "***@***.com" #接收人

## 发送邮件
s = smtplib.SMTP() #实例化对象
s.connect(mail_host, 25) #连接163邮箱服务器,端口号为25
s.login(mail_sender, mail_passwd) #登录邮箱
s.sendmail(mail_sender, ["***@***.com"], msg.as_string())
s.quit()

运行上面的脚本,收到的邮件如下:

在这里插入图片描述
构造邮件内容的MIMEText类有三个参数,第一个参数为文本内容,第二个参数‘plain’设置文本格式,第三个参数设置编码格式。

带图片邮件

接下来我们发送一封带有图片的邮件,代码如下:

from email.mime.text import MIMEText
from email.mime.multipart import MIMEMultipart
import smtplib
from email.mime.image import MIMEImage

mail_host = "smtp.163.com" #SMTP服务器地址
mail_sender = "***@163.com" #账号
mail_passwd = "****" #密码

msg = MIMEMultipart('related')
msg["Subject"] = "这里是邮件主题"
msg["From"] = mail_sender #发送人
msg["To"] = "songzp@***.com" #接收人

#html格式的邮件正文
content = '''
<body>
<p>测试Python发送带图片的邮件...</p>
<p>图片如下:</p>
<p><img src="cid:testimage" alt="testimage"></p>
</body>
'''
msg.attach(MIMEText(content,'html','utf-8'))

#读取图片
fp = open('testimage.jpg', 'rb')  #打开文件
msgImage = MIMEImage(fp.read()) #读入 msgImage 中
fp.close() #关闭文件

# 定义图片 ID,在 HTML 文本中引用
msgImage.add_header('Content-ID', 'testimage') 
msg.attach(msgImage)

## 发送邮件
s = smtplib.SMTP() #实例化对象
s.connect(mail_host, 25) #连接163邮箱服务器,端口号为25
s.login(mail_sender, mail_passwd) #登录邮箱
s.sendmail(mail_sender, ["songzp@***.com"], msg.as_string())
s.quit()

邮件的正文为html格式,在img标签的src属性中给图片设置一个cid,用MIMEImage读取图片二进制流,再attach进邮件中。收到的邮件如下:

在这里插入图片描述

带表格邮件

接下来,我们在邮件内容里面添加表格,代码如下:

from email.mime.text import MIMEText
from email.mime.multipart import MIMEMultipart
import smtplib
from email.mime.image import MIMEImage

mail_host = "smtp.163.com" #SMTP服务器地址
mail_sender = "***@163.com" #账号
mail_passwd = "****" #密码

msg = MIMEMultipart('related')
msg["Subject"] = "带有表格的邮件"
msg["From"] = mail_sender #发送人
msg["To"] = "songzp@***.com" #接收人

#html格式的邮件正文
content = '''
<html>
<body>
<p><strong>这是加粗字体,可以是一个标题</strong></p>
<p>下面是一个表格,表格格式可以随意设置</p>
<table width="500" bordercolor="black" border="1" cellspacing="0">
 <tr>
  <td><strong>列名1</strong></td>
  <td><strong>列名2</strong></td>
  <td><strong>列名3</strong></td>
 </tr>
 <tr>
  <td>str1</td>
  <td>10</td>
  <td>2019-01-01</td>
 </tr>
 <tr>
  <td>str2</td>
  <td>20</td>
  <td>2019-01-02</td>
 </tr>
 <tr>
  <td>str3</td>
  <td>30</td>
  <td>2019-01-03</td>
 </tr>
</table>
</body>
</html>
'''
msg.attach(MIMEText(content,'html','utf-8'))

## 发送邮件
s = smtplib.SMTP() #实例化对象
s.connect(mail_host, 25) #连接163邮箱服务器,端口号为25
s.login(mail_sender, mail_passwd) #登录邮箱
s.sendmail(mail_sender, ["songzp@***.com"], msg.as_string())
s.quit()

收到的邮件如下:

在这里插入图片描述
邮件中添加表格比较简单,只需要编辑HTML格式的邮件正文就行。标签table之间的内容就是表格,tr标签之间的内容是表格的一行,td标签表示一个单元格,表格的边框、颜色等格式都可以通过标签属性设置。在实际工作中,表格可能有多行多列,或者不固定,可以根据自己的应用场景通过循环等编辑HTML格式的文本。

带有附件的邮件

最后一个常用的场景就是发送带有附件的邮件,代码如下:

from email.mime.text import MIMEText
from email.mime.multipart import MIMEMultipart
import smtplib
from email.mime.image import MIMEImage

mail_host = "smtp.163.com" #SMTP服务器地址
mail_sender = "***@163.com" #账号
mail_passwd = "****" #密码

msg = MIMEMultipart()
msg["Subject"] = "带有附件的邮件"
msg["From"] = mail_sender #发送人
msg["To"] = "songzp@***.com" #接收人

# 邮件正文
content = '''
这是一封带有附件的邮件...
有两个附件
'''
msg.attach(MIMEText(content,'plain','utf-8'))

# 构造附件1,txt文件
att1 = MIMEText(open('test.txt', 'rb').read(), 'base64', 'utf-8')
att1["Content-Type"] = 'application/octet-stream'
# 这里的filename可以任意写,写什么名字,邮件中显示什么名字
att1["Content-Disposition"] = 'attachment; filename="test.txt"'
msg.attach(att1)

# 构造附件2,xlsx文件
att2 = MIMEText(open('test.xlsx', 'rb').read(), 'base64', 'utf-8')
att2["Content-Type"] = 'application/octet-stream'
att2["Content-Disposition"] = 'attachment; filename="test.xlsx"'
msg.attach(att2)

## 发送邮件
s = smtplib.SMTP() #实例化对象
s.connect(mail_host, 25) #连接163邮箱服务器,端口号为25
s.login(mail_sender, mail_passwd) #登录邮箱
s.sendmail(mail_sender, ["songzp@***.com"], msg.as_string())
s.quit()

收到的邮件如下:

在这里插入图片描述
原理都是一样的,邮件中需要加的内容都要attach进msg中,最后的sendemail方法只有三个参数:谁发的邮件,发给谁,发什么

如果你经常需要开发自动发邮件的脚本,你可以将上述代码写成一个函数,或一个类(class),比如下面的函数:

def send_mail(mail_host, mail_sender, mail_pass, to_list, sub, content, attach_list=[], _subtype='plain'):
    #msg = MIMEMultipart(content,_charset='utf-8') 
    msg = MIMEMultipart()
    msg['Subject'] = sub 
    msg['From'] = mail_sender 
    msg['To'] = to_list 
    try: 
        msg.attach(MIMEText(content, _subtype, 'utf-8'))
        s = smtplib.SMTP() 
        s.connect(mail_host, 25) 
        s.login(mail_sender, mail_pass) 

        if attach_list:
            for att_path in attach_list:
                path_arr = att_path.split(os.path.sep)
                file_name = path_arr[len(path_arr) - 1]
                att = MIMEText(open(att_path, 'rb').read(), 'base64', 'utf-8')
                att["Content-Type"] = 'application/octet-stream'
                att["Content-Disposition"] = 'attachment; filename="%s"' % file_name
                msg.attach(att)

        s.sendmail(mail_sender, to_list.split(u","), msg.as_string()) 
        s.close() 
        return True
    except Exception as e:
        print e 
        return False

这个自定义函数除了不能用来发送带图片的邮件,其他的都可以使用。自己动手写一个全能的发邮件函数吧

  • 6
    点赞
  • 78
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

追光的鲲

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值