python学习笔记(1)- 利用SMPT 实现自动发送邮件脚本
python学习笔记(1)- 利用SMPT 实现自动发送邮件脚本,读取txt和excel邮箱列表
起因
学校正好组织夏令营,需要邮件通知参与夏令营的同学,报名信息中包含手机号和邮件(运营费问题,放弃了手机)。然后,有小伙伴提出python发送邮件的方式,参考了几篇博客之后,有了此文章。
涉及内容
python 读取txt文本
参考链接:https://blog.csdn.net/qq_37828488/article/details/100024924
python 使用openpyxl读取Excel表格
参考链接:https://blog.csdn.net/weixin_43094965/article/details/82226263
python 使用内置SMTP模块
原理
SMTP是一个相对简单的基于文本的协议。在其之上指定了一条消息的一个或多个接收者(在大多数情况下被确认是存在的),然后消息文本会被传输。可以很简单地通过telnet程序来测试一个SMTP服务器。SMTP使用TCP端口25。要为一个给定的域名决定一个SMTP服务器,需要使用MX (Mail eXchange)DNS。(节自——SMTP百度百科https://baike.baidu.com/item/SMTP/175887?fr=aladdin)
Python内置SMTP模块
Python内置对SMTP的支持,可以发送纯文本邮件、HTML邮件以及带附件的邮件
参考链接1:https://www.liaoxuefeng.com/wiki/897692888725344/923057144964288
参考链接2:
https://www.runoob.com/python/python-email.html
python创建SMTP对象的语法如下:
import smtplib
smtpObj = smtplib.SMTP( [host [, port [, local_hostname]]] )
smtpObj.connect(smtpserver,25) #(缺省)默认端口是25 也可以根据服务器进行设定
smtpObj.login(user,password) #登陆smtp服务器
参数说明:
- host: SMTP 服务器主机。 你可以指定主机的ip地址或者域名如: runoob.com,这个是可选参数。
- port: 如果你提供了 host 参数, 你需要指定 SMTP 服务使用的端口号,一般情况下 SMTP 端口号为25。
- local_hostname: 如果 SMTP 在你的本机上,你只需要指定服务器地址为 localhost 即可。
- smtserver 发件人所用邮箱的SMPT服务器地址
- user,password是发件人的邮箱和密码
Python SMTP 对象使用 sendmail 方法发送邮件,语法如下:
SMTP.sendmail(from_addr, to_addrs, msg[, mail_options, rcpt_options])
参数说明:
- from_addr: 邮件发送者地址。
- to_addrs: 字符串列表,邮件发送地址。
- msg: 发送消息,实质为MIMEText对象
- 这里要注意一下第三个参数,msg 是字符串,表示邮件。我们知道邮件一般由标题,发信人,收件人,邮件内容,附件等构成,发送邮件的时候,要注意 msg 的格式。这个格式就是 smtp 协议中定义的格式。
MIMEText对象的语法如下:
import smtplib
from email.mime.text import MIMEText
from email.header import Header
msg = MIMEText('您好,巴拉巴拉巴拉!', 'plain', 'utf-8') #创建纯文本内容
msg['From'] = Header('LuH1124的CSDN博客','utf-8')
msg['To'] = Header('张三','utf-8')
subject = '小小的 SMTP 测试邮件'
msg['Subject'] = Header(subject, 'utf-8')
参数说明:
- 第五行第二个参数是MIME的subtype,传入’plain’,最终的MIME就是’text/plain’,最后一定要用utf-8编码保证多语言兼容性。
- 第六行为 发件人姓名
- 第七行为 收件人姓名
- 第八、九行为 邮件主题
SMPT发送邮件流程
SMTP通讯的基本流程可以概括为以下几点:
1.连接SMTP服务器
2.登陆用户名和密码
3.发送指定邮件内容
4.退出SMTP连接
发送纯文本参考程序如下:
# coding:utf-8
import importlib
import sys
importlib.reload(sys)
sys.setdefaultencoding('utf8')
import smtplib
from smtplib import SMTP
from email.mime.text import MIMEText
from email.header import Header
#构造纯文本邮件内容
msg = MIMEText('您好,巴拉巴拉巴拉!','plain','utf-8')
msg['From'] = Header('LuH1124的CSDN博客','utf-8')
msg['To'] = Header('张三','utf-8')
subject = '小小的 SMTP 测试邮件'
msg['Subject'] = Header(subject, 'utf-8')
#发送者邮箱
sender = 'aaa@qq.com'
#发送者的登陆用户名和密码
user = 'aaa@qq.com'
password = 'xxxxxx'
#发送者邮箱的SMTP服务器地址 比如 SMTP.163.com
smtpserver = 'xxxx'
#接收者的邮箱地址
receiver = ['aaa@qq.com','aaa@qq.com'] #receiver 可以是一个list
smtp = smtplib.SMTP() #实例化SMTP对象
smtp.connect(smtpserver,25) #(缺省)默认端口是25 也可以根据服务器进行设定
smtp.login(user,password) #登陆smtp服务器
smtp.sendmail(sender,receiver,msg.as_string()) #发送邮件
smtp.quit() # 退出SMTP服务器
测试结果如下:
发送附件参考程序如下:
发送带附件的邮件,首先要创建MIMEMultipart()实例,然后构造附件,如果有多个附件,可依次构造,最后利用smtplib.smtp发送。
# -*- coding: UTF-8 -*-
import smtplib
from email.mime.text import MIMEText
from email.mime.multipart import MIMEMultipart
from email.header import Header
sender ='aaa@qq.com'
receivers = ['aaa@qq.com'] # 接收邮件,可设置为你的QQ邮箱或者其他邮箱
#发送者的登陆用户名和密码
user = 'aaa@qq.com'
password = 'buzhiDAO99'
#发送者邮箱的SMTP服务器地址 比如 SMTP.163.com
smtpserver = 'smtp.buaa.edu.cn'
#创建一个带附件的实例
msg = MIMEMultipart()
msg['From'] = Header('LuH1124的CSDN博客','utf-8')
msg['To'] = Header('张三','utf-8')
subject = '带附件的 SMTP 测试邮件'
msg['Subject'] = Header(subject, 'utf-8')
#邮件正文内容
msg.attach(MIMEText('附件发送测试……', 'plain', 'utf-8'))
# 构造附件1,传送当前目录下的 test.txt 文件
att1 = MIMEText(open('test.png', 'rb').read(), 'base64', 'utf-8')
att1["Content-Type"] = 'application/octet-stream'
# 这里的filename可以任意写,写什么名字,邮件中显示什么名字
att1["Content-Disposition"] = 'attachment; filename="test.png"'
msg.attach(att1)
# 构造附件2,传送当前目录下的 runoob.txt 文件
att2 = MIMEText(open('email.txt', 'rb').read(), 'base64', 'utf-8')
att2["Content-Type"] = 'application/octet-stream'
att2["Content-Disposition"] = 'attachment; filename="email.txt"'
msg.attach(att2)
try:
smtp = smtplib.SMTP()
smtp.connect(smtpserver,25) #(缺省)默认端口是25 也可以根据服务器进行设定
smtp.login(user,password) #登陆smtp服务器
smtp.sendmail(sender, receivers, msg.as_string())
print("邮件发送成功")
smtp.quit() # 退出SMTP服务器
except smtplib.SMTPException:
print("Error: 无法发送邮件")
测试结果如下:
读取txt和Excel中保存的邮箱信息
以上是自动发邮件的操作,下面讲一下从txt和Excel中读取邮箱信息,并添加到邮件正文、修改 收件人列表 的程序。
- 首先是读取txt文本实现群发,为方便操作,txt文本的内容如图所示:
txt文本中,直接从excel中复制邮箱列即为上述格式。主要用到的函数为readlines(),见文章开头参考链接。每个邮箱字符串中之间存在一个换行符“\n”,需要使用字符串的 str.split()方法
定义函数如下:
def read_txt(path):
email_list =[]
with open(path, "r") as file:
for line in file.readlines():
#ine = line.strip('\n')
email_list.append(line)
return email_list
函数返回值既可作为 receiver这个list的value,带入程序即可。path参数指文件的路径与文件名。
- 其次是读取Excel文本,Excel文本的内容格式如图所示:
此文本中我们想要实现对姓名(可在邮箱正文内容中 添加同学的姓名)和邮箱的的提取,并分别保存在name_list和email_list中。
import openpyxl
wb = openpyxl.load_workbook("Chat_Sum.xlsx")
ws = wb.active # 获取sheet1
name_range = ws['B2':'B' + str(ws.max_row)]
email_range = ws['E2':'E' + str(ws.max_row)]
#print(ws.max_row)
#print(ws.max_column)
name_list = []
email_list = []
#name_range
for name in name_range:
name_list.append(name[0].value)
for email in email_range:
email_list.append(email[0].value)
引入openpyxl模块,使用load_workbook读取excel,使用active 读取sheet1,使用max_row 读取表格行数,选取B2列和E2列数据,读取后分别保存在name_range和email_range当中,数据结构为 tuple元组列表,索引【0】读取。
最后程序,可实现单发与群发,更改正文名字
# coding:utf-8
import importlib
import sys
importlib.reload(sys)
import smtplib
from smtplib import SMTP
from email.mime.text import MIMEText
from email.header import Header
import openpyxl
def read_xlsx(path):
wb = openpyxl.load_workbook(path)
ws = wb.active # 获取sheet1
name_range = ws['B2':'B' + str(ws.max_row)]
email_range = ws['E2':'E' + str(ws.max_row)]
#print(ws.max_row)
#print(ws.max_column)
name_list = []
email_list = []
#name_range
for name in name_range:
name_list.append(name[0].value)
for email in email_range:
email_list.append(email[0].value)
return name_list,email_list
def read_txt(path):
email_list =[]
with open(path, "r") as file:
for line in file.readlines():
#ine = line.strip('\n')
email_list.append(line)
return email_list
def init_Smpt(smtpserver,user,password ):
smtp = smtplib.SMTP() # 实例化SMTP对象
smtp.connect(smtpserver, 25) # (缺省)默认端口是25 也可以根据服务器进行设定
smtp.login(user, password) # 登陆smtp服务器,login()方法用来登录SMTP服务器
return smtp
def send_Mass_Smpt(sender, receiver_list, msg): # 群发相同内容
smtp.sendmail(sender,receiver_list, msg.as_string()) # 发送邮件 ,这里有三个参数
'''
sendmail()方法就是发邮件,由于可以一次发给多个人,所以传入一个list,邮件正文
是一个str,as_string()把MIMEText对象变成str。
'''
smtp.quit()
def send_Alone_Smpt(sender, receiver_list, msg): # 单发不同同内容
smtp.sendmail(sender,receiver_list, msg.as_string()) # 发送邮件 ,这里有三个参数
'''
sendmail()方法就是发邮件,由于可以一次发给多个人,所以传入一个list,邮件正文
是一个str,as_string()把MIMEText对象变成str。
'''
smtp.quit()
if __name__ == '__main__':
# 文件地址
txt_path = "./email.txt"
xlsx_path = "./Chat_Sum.xlsx"
# 接收者的姓名和邮箱地址
receiver_name_list, receiver_list = read_xlsx(xlsx_path) # receiver 可以是一个list
flag = False
'''
**群发标志位 True:可读取 txt 和 xlsx文件
False:只可读取 xlsx文件**
'''
# 发送者邮箱
sender = 'aaa@qq.com'
# 发送者的登陆用户名和密码
user = 'aaa@qq.com'
password = '*******' # 密码需要修改成自己的
# 发送者邮箱的SMTP服务器地址
smtpserver = 'smtp.163.com'
if flag:
# 群发
# 构建邮件邮件内容
smtp = init_Smpt(smtpserver,user,password)
msg = MIMEText('您好,欢迎~~~~~!', 'plain', 'utf-8')
# 构造纯文本邮件内容,第12行是发送人,第3行是收件人,第4行是邮件标题
msg['From'] = Header('CSDnTEST','utf-8')
msg['To'] = Header('张三','utf-8')
subject = ' SMTP 测试邮件'
msg['Subject'] = Header(subject, 'utf-8')
send_Mass_Smpt(sender, receiver_list, msg)
else:
# 单发
# 构建邮件邮件内容,
for index in range(len(receiver_name_list)):
smtp = init_Smpt(smtpserver,user,password) # 需重新初始化邮箱
msg = MIMEText('{}您好,欢迎参加~~~~夏令营!'.format(receiver_name_list[index]), 'plain', 'utf-8') #format 格式化输出姓名
# 构造纯文本邮件内容,第12行是发送人,第3行是收件人,第4行是邮件标题
msg['From'] = Header('CSDNtest','utf-8')
msg['To'] = Header('张三','utf-8')
subject = 'SMTP 测试邮件'
msg['Subject'] = Header(subject, 'utf-8')
send_Alone_Smpt(sender, receiver_list[index], msg)