Python-实现邮件发送:flask框架或django框架可以直接使用
在项目中,会使用到发送邮件的功能。不同框架的配置可能有所不同,直接写一个不依赖框架配置的邮件发送模块。
使用的功能:
1、可以发送给多个邮箱
2、可以实现抄送多个邮箱
3、可以添加多个文件附件
一、不使用多线程
import smtplib
from email.mime.text import MIMEText
from email.mime.multipart import MIMEMultipart
from email.mime.base import MIMEBase
from email import encoders
from email.header import Header
from email.utils import formataddr
from threading import Threadclass Email:EMAIL_HOST = 'smtp.qq.com' # 如果是 163 改成 smtp.163.comEMAIL_PORT = 587 # qq邮箱服务的端口:465、587EMAIL_HOST_USER = "xxx@qq.com" # 发送邮件的邮箱帐号EMAIL_HOST_PASSWORD = "xxx" # 授权码,各邮箱的设置中启用smtp服务时获取FROM_EMAIL_USER = EMAIL_HOST_USER # 收件人显示发件人的邮箱FROM_EMAIL_NAME = '广州市xxxx技术有限公司' # 设置发件人的名字,在项目使用,一般是用公司名字def __init__(self, receiver_email, subject, message=None, html_message=None, file_path=None,cc_email=None):self.receiver_email = receiver_email # 接收人邮件self.cc_email = cc_email #邮件抄送人self.subject = subject # 邮件的主题self.message = message # 邮件文本内容self.html_message = html_message # 邮件的html内容 注意:文本内容与html同时有的时候,html覆盖文本内容self.file_path = file_path # 附件文件的路径,str 或 [str,]self.email = MIMEMultipart() #创建一个邮件对象,发送的邮件的信息都设置到这个对象中#接收人邮件:必须有if type(receiver_email) in [list,str,tuple]:if type(receiver_email) in [list,tuple]:self.receiver_email = ','.join(receiver_email)else:raise Exception('邮件抄送人必须是字符串、列表或元组形式')#抄送人邮件: 可选if cc_email:if type(cc_email) in [list,str,tuple]:if type(cc_email) in [list,tuple]:self.cc_email = ','.join(cc_email)else:raise Exception('邮件抄送人必须是字符串、列表或元组形式')def start(self):# 1、设置邮箱对象的发送人、接收人和主题self.email['From'] = formataddr((self.FROM_EMAIL_NAME, self.FROM_EMAIL_USER))self.email['To'] = self.receiver_emailself.email['Subject'] = Header(self.subject, 'utf-8')self.email['Cc'] = self.cc_email# 2、设置内容,如果同时attach了html信息和文本信息,文本信息会被转成附件文件。if not self.message and not self.html_message:raise Exception('发送的邮件每月携带任何内容...')if self.html_message:self.email.attach(MIMEText(self.html_message, 'html', 'utf-8'))else:self.email.attach(MIMEText(self.message, 'plain', 'utf-8'))# 3、邮箱的附件文件if self.file_path != None:if isinstance(self.file_path, str):self.file_path = [self.file_path]elif isinstance(self.file_path, list):passelse:raise Exception('邮件携带的附件,格式是文件字符串路径,就列表套文件字符串路径,不能是其他格式')for path in self.file_path:with open(path, "rb") as attachment:part = MIMEBase("application", "octet-stream")part.set_payload(attachment.read())encoders.encode_base64(part)part.add_header("Content-Disposition",f"attachment; filename= {path}",)self.email.attach(part)# 4、连接smtp服务器,发送邮件with smtplib.SMTP(self.EMAIL_HOST, self.EMAIL_PORT) as server:# QQ邮箱的服务器域名和端口server.starttls() # 使用 TLS 加密连接# 使用qq邮箱登录:邮箱号和授权码server.login(self.EMAIL_HOST_USER, self.EMAIL_HOST_PASSWORD)# 发送邮件,发送的是邮箱对象server.send_message(self.email)if __name__ == '__main__':receiver_email = ['xxx@163.com','yyy@qq.com'] #接收人邮件账号cc_email = ['xxxx@163.com','yyy@qq.com'] #抄送人的邮件账号subject = '登录验证码' #主题message = '您的验证码是234523,10分钟中内有效,若非本人操作请忽视...' #文本信息file_path = ['1.txt', '2.txt'] #同目录下创建1.txt和2.txt 文件,测试附件文件html_message = '<h1>登录验证码,5分钟内有效</h1><p>您的验证码是:546783</p><p>注意:若非本人操作,建议删除邮件防止泄漏验证码信息</p>' #html信息,有这个使用这个不使用文本信息#非多线程发送email_obj_2 = Email(receiver_email=receiver_email,subject=subject,message=message,file_path=file_path,html_message=html_message,cc_email=cc_email)email_obj_2.start()
二、使用线程封装
import smtplib
from email.mime.text import MIMEText
from email.mime.multipart import MIMEMultipart
from email.mime.base import MIMEBase
from email import encoders
from email.header import Header
from email.utils import formataddr
from threading import Threadclass ThreadEmail(Thread):EMAIL_HOST = 'smtp.qq.com' # 如果是 163 改成 smtp.163.comEMAIL_PORT = 587 # qq邮箱服务的端口:465、587EMAIL_HOST_USER = "xxx@qq.com" # 发送邮件的邮箱帐号EMAIL_HOST_PASSWORD = "xxx" # 授权码,各邮箱的设置中启用smtp服务时获取FROM_EMAIL_USER = EMAIL_HOST_USER # 收件人显示发件人的邮箱FROM_EMAIL_NAME = '广州市xxxx技术有限公司'#设置发件人的名字,在项目使用,一般是用公司名字def __init__(self, receiver_email, subject, message=None, html_message=None, file_path=None,cc_email=None):super().__init__()self.receiver_email = receiver_email # 接收人邮件self.cc_email = cc_email # 邮件抄送人self.subject = subject # 邮件的主题self.message = message # 邮件文本内容self.html_message = html_message # 邮件的html内容 注意:文本内容与html同时有的时候,html覆盖文本内容self.file_path = file_path # 附件文件的路径,str 或 [str,]self.email = MIMEMultipart() # 创建一个邮件对象,给这个对象添加 邮件需要的各种信息# 接收人邮件:必须有if type(receiver_email) in [list, str, tuple]:if type(receiver_email) in [list, tuple]:self.receiver_email = ','.join(receiver_email)else:raise Exception('邮件抄送人必须是字符串、列表或元组形式')# 抄送人邮件: 可选if cc_email:if type(cc_email) in [list, str, tuple]:if type(cc_email) in [list, tuple]:self.cc_email = ','.join(cc_email)else:raise Exception('邮件抄送人必须是字符串、列表或元组形式')def run(self) -> None:# 1、设置邮箱对象的发送人、接收人和主题self.email['From'] = formataddr((self.FROM_EMAIL_NAME, self.FROM_EMAIL_USER))self.email['To'] = self.receiver_emailself.email['Subject'] = Header(self.subject, 'utf-8')self.email['Cc'] = self.cc_email# 2、设置内容,如果同时attach了html信息和文本信息,文本信息会被转成附件文件。if not self.message and not self.html_message:raise Exception('发送的邮件每月携带任何内容...')if self.html_message:self.email.attach(MIMEText(self.html_message, 'html', 'utf-8'))else:self.email.attach(MIMEText(self.message, 'plain', 'utf-8'))# 3、邮箱的附件文件if self.file_path != None:if isinstance(self.file_path, str):self.file_path = [self.file_path]elif isinstance(self.file_path, list):passelse:raise Exception('邮件携带的附件,格式是文件字符串路径,就列表套文件字符串路径,不能是其他格式')for path in self.file_path:with open(path, "rb") as attachment:part = MIMEBase("application", "octet-stream")part.set_payload(attachment.read())encoders.encode_base64(part)part.add_header("Content-Disposition",f"attachment; filename= {path}",)self.email.attach(part)# 4、连接smtp服务器,发送邮件with smtplib.SMTP(self.EMAIL_HOST, self.EMAIL_PORT) as server:# QQ邮箱的服务器域名和端口server.starttls() # 使用 TLS 加密连接# 使用qq邮箱登录:邮箱号和授权码server.login(self.EMAIL_HOST_USER, self.EMAIL_HOST_PASSWORD)# 发送邮件,发送的是邮箱对象server.send_message(self.email)if __name__ == '__main__':receiver_email = ['xxx@163.com','yyy@qq.com'] #接收人邮件账号cc_email = ['xxxx@163.com','yyy@qq.com'] #抄送人的邮件账号subject = '登录验证码' #主题message = '您的验证码是234523,10分钟中内有效,若非本人操作请忽视...' #文本信息file_path = ['1.txt', '2.txt'] #同目录下创建1.txt和2.txt 文件,测试附件文件html_message = '<h1>登录验证码,5分钟内有效</h1><p>您的验证码是:546783</p><p>注意:若非本人操作,建议删除邮件防止泄漏验证码信息</p>' #html信息,有这个使用这个不使用文本信息#多线程发送email_obj = ThreadEmail(receiver_email=receiver_email,subject=subject,message=message,file_path=file_path,html_message=html_message,cc_email=cc_email)email_obj.start()
三、通用性
1、对于脚本代码,也可以直接使用
2、在框架中,也可以直接使用
相关文章:
Python-实现邮件发送:flask框架或django框架可以直接使用
在项目中,会使用到发送邮件的功能。不同框架的配置可能有所不同,直接写一个不依赖框架配置的邮件发送模块。 使用的功能: 1、可以发送给多个邮箱 2、可以实现抄送多个邮箱 3、可以添加多个文件附件 一、不使用多线程 import smtplib from…...
使用亚马逊云科技Amazon SageMaker,为营销活动制作广告素材
广告公司可以使用生成式人工智能和文字转图像根基模型,制作创新的广告素材和内容。在本篇文案中,将演示如何使用亚马逊云科技Amazon SageMaker从现有的基本图像生成新图像,这是一项完全托管式服务,用于大规模构建、训练和部署机器…...
conda环境安装opencv带cuda版本
主要是cmake编译选项需要修改 以下两个选项按照自己情况修改 -D OPENCV_EXTRA_MODULES_PATH../opencv_contrib/modules \ -D CUDA_TOOLKIT_ROOT_DIR/usr/local/cuda-12.2 \ 其中/home/lixin/anaconda3/envs/stereo 改成你自己的conda环境 cmake -D CMAKE_BUILD_TYPER…...
R语言中的数据结构----矩阵
目录 (1)创建矩阵 (2) 线性代数运算 (3)矩阵索引 (4)矩阵元素的筛选 (5)增加或删除矩阵的行或列 (6)apply()函数 (…...
Llama-2 推理和微调的硬件要求总结:RTX 3080 就可以微调最小模型
大语言模型微调是指对已经预训练的大型语言模型(例如Llama-2,Falcon等)进行额外的训练,以使其适应特定任务或领域的需求。微调通常需要大量的计算资源,但是通过量化和Lora等方法,我们也可以在消费级的GPU上…...
C++多线程的用法(包含线程池小项目)
一些小tips: 编译命令如下: g 7.thread_pool.cpp -lpthread 查看运行时间: time ./a.out 获得本进程的进程id: this_thread::get_id() 需要引入的库函数有: #include<thread> // 引入线程库 #include<mutex> //…...
react ant ice3 实现点击一级菜单自动打开它下面最深的第一个子菜单
1.问题 默认的如果没有你的菜单结构是这样的: [{children: [{name: "通用配置"parentId: "1744857774620672"path: "basic"}],name: "系统管理"parentId: "-1"path: "system"} ]可以看到每层菜单的p…...
关于 Qt串口不同电脑出现不同串口号打开失败 的解决方法
若该文为原创文章,转载请注明原文出处 本文章博客地址:https://hpzwl.blog.csdn.net/article/details/132842297 红胖子(红模仿)的博文大全:开发技术集合(包含Qt实用技术、树莓派、三维、OpenCV、OpenGL、ffmpeg、OSG、单片机、软…...
可观测性在灰度发布中的应用
前言 随着云计算的发展、云原生时代的来临,企业数字化转型进程不断深入,应用开发也越来越多地基于微服务化模式,快速迭代的能力使得应用开发更高效、更灵活。同时,也不得不面临应用版本快速升级所带来的的巨大挑战。 传统的发布方…...
vscode开发油猴插件环境配置指南
文章目录 一、环境配置1.1油猴插件开始编写代码1.2油猴插件配置1.2.1浏览器插件权限1.2.2插件自身权限 2. 油猴脚本API学习2.1 头文件2.2 油猴API 一、环境配置 1.1油猴插件开始编写代码 在vscode 中写入如下代码‘ // UserScript // name cds_test // namespace …...
网站不收录没排名降权怎么处理-紧急措施可恢复网站
网站降权对于SEO人员来说是非常致命的打击,因为网站一旦被搜索引擎降权,排名会严重地下降,网站的流量也会大幅下降,直接影响到收益。而且处理不好的话会导致恢复的时间周期无限拉长,所以网站被降权后我们要第一时间采取…...
C++vector模拟实现
vector模拟实现 1.构造函数2.拷贝构造3.析构赋值运算符重载4.iterator5.modifiers5.1push_back5.2pop_back5.3empty5.4insert5.5erase5.6swap 6.Capacity6.1size6.2capacity6.3reserve6.4resize6.5empty 7.Element access7.1operator[]7.2at 8.在谈reserve vector官方库实现的是…...
《DATASET DISTILLATION》
这篇文章提出了数据浓缩的办法,在前面已有的知识浓缩(压缩模型)的经验上,提出了不压缩模型,转而压缩数据集的办法,在压缩数据集上训练模型得到的效果尽可能地接近原始数据集的效果。 摘要 模型蒸馏的目的是…...
GDPU 数据结构 天码行空1
1. 病历信息管理 实现病历查询功能。具体要求如下: 定义一个结构体描述病人病历信息(病历号,姓名,症状);完成功能如下: 输入功能:输入5个病人的信息; 查询功能:输入姓名,在5个病历中进行查找,如果找到则显示该人的信息,…...
【C++】红黑树的模拟实现
🌇个人主页:平凡的小苏 📚学习格言:命运给你一个低的起点,是想看你精彩的翻盘,而不是让你自甘堕落,脚下的路虽然难走,但我还能走,比起向阳而生,我更想尝试逆风…...
【多线程】Thread 类 详解
Thread 类 详解 一. 创建线程1. 继承 Thread 类2. 实现 Runnable 接口3. 其他变形4. 多线程的优势-增加运行速度 二. Thread 类1. 构造方法2. 常见属性3. 启动线程-start()4. 中断线程-interrupt()5. 线程等待-join()6. 线程休眠-sleep()7. 获取当前线程引用 三. 线程的状态1. …...
LINUX 网络管理
目录 一、NetworkManager的特点 二、配置网络 1、使用ip命令临时配置 1)查看网卡在网络层的配置信息 2)查看网卡在数据链路层的配置信息 3)添加或者删除临时的网卡 4)禁用和启动指定网卡 2、修改配置文件 3、nmcli命令行…...
refresh rate
1920 x 1080 显卡刷新率 60...
使用 NGINX Unit 实施应用隔离
原文作者:Artem Konev - Senior Technical Writer 原文链接:使用 NGINX Unit 实施应用隔离 转载来源:NGINX 中文官网 NGINX 唯一中文官方社区 ,尽在 nginx.org.cn NGINX Unit 特性集的最新动态之一是支持应用隔离,该特…...
2023/09/12 qtc++
实现一个图形类(Shape) ,包含受保护成员属性:周长、面积, 公共成员函数:特殊成员函数书写 定义一个圆形类(Circle) ,继承自图形类,包含私有属性:半径 公共成员函数:特殊成员函数…...
全科医学科常用评估量表汇总,建议收藏!
根据全科医学科医生的量表使用情况,笔者整理了10个常用的全科医学科量表,可在线评测直接出结果,可转发使用,可生成二维码使用,可创建项目进行数据管理,有需要的小伙伴赶紧收藏! 日常生活能力量表…...
了解消息中间件的基础知识
为什么要使用消息中间件? 解耦:消息中间件可以使不同的应用程序通过解耦的方式进行通信,减少系统间的依赖关系提供异步通信:消息中间件可以实现异步消息传递,提高系统的响应性能。流量削峰:消息中间件可以…...
【linux】Linux wps字体缺失、加粗乱码解决
解决wps字体缺失问题 1、下载字体包 git clone https://github.com/iamdh4/ttf-wps-fonts.git2、创建单独放置字体的目录 mkdir /usr/share/fonts/wps-fonts3、复制字体到系统目录下 cp ttf-wps-fonts/* /usr/share/fonts/wps-fonts4、修改字体权限 chmod 644 /usr/share/f…...
每日两题 103二叉树的锯齿形层序遍历(数组) 513找树左下角的值(队列)
103 题目 103 给你二叉树的根节点 root ,返回其节点值的 锯齿形层序遍历 。(即先从左往右,再从右往左进行下一层遍历,以此类推,层与层之间交替进行)。 示例 1: 输入:root [3,9,…...
ROS2报错:ImportError: cannot import name ‘Log‘ from ‘rosgraph_msgs.msg‘
在使用ros2的bag命令查看数据集信息时报错 Traceback (most recent call last):File "/opt/ros/noetic/bin/rosbag", line 34, in <module>import rosbagFile "/opt/ros/noetic/lib/python3/dist-packages/rosbag/__init__.py", line 33, in <mo…...
【Vue】Vue中的代码分为哪几种类型?
在 Vue 中的代码可以分为以下几种类型: 1.模板代码 模板代码是 Vue 中用来生成 HTML 的一种语法,可以通过 Vue 的模板语法和指令来动态渲染页面。模板代码一般写在 Vue 组件的 template 标签中。 2.JavaScript 代码 JavaScript 代码是 Vue 组件中用来…...
es6中includes用法
js中的includes用法 1.数组 includes 可以判断一个数组中是否包含某一个元素,并返回true 或者false [a,b,c].includes(a) true [a,b,c].includes(1) false includes可以包含两个参数,第二个参数表示判断的起始位置 起始位置第一个数字是0。 2.字符串 …...
QT中QRadioButton实现分组C++
通过对QRadioButton组件进行分组可解决QRadioButton组件的互斥性 实现如下。 假设已设计好UI并且有UI代码情况: 头文件引用: #include <QButtonGroup> 分组功能 ,cpp文件代码实现: Your_Project::Your_Project(QWidge…...
kafka实战报错解决问题
需求 在一个在线商城中,用户下单后需要进行订单的处理。为了提高订单处理的效率和可靠性,我们使用Kafka来实现订单消息的异步处理。当用户下单后,订单信息会被发送到Kafka的一个Topic中,然后订单处理系统会从该Topic中消费订单消…...
vite+react 使用 react-activation 实现缓存页面
对应的版本 "react": "^18.2.0", "react-activation": "^0.12.4", "react-dom": "^18.2.0", "react-router-dom": "^6.15.0",react-activation 这是一个npm包,在react keep alive…...
专门做化妆品的网站/优化工作流程
2.1 问题 如图配置IPv6地址和OSPFv3区域 查看OSPFv3邻居和数据库LSA 确保 R1 和 R3 可以互相访问2.2 方案 搭建实验环境,如图-2所示。 图-2 2.3 步骤 实现此案例需要按照如下步骤进行。 1)配置R1 地址,并启用 OSPFv3 <Huawei>und…...
滕州网站建设 助企网络/上海关键词优化按天计费
在系统开机后我们的桌面图标却不翼而飞了,这该怎么办?没有桌面图标就相当于我们进房间没有了门,就这样无路可走了吗?别慌,下文就给出了让你的桌面图标重现的方法。 右键单击桌面,排列桌面图标,显…...
深圳物流公司网站/小红书推广方式
链接静态C库和C 代码时出现“未定义的引用”错误我有一个测试文件(仅用于链接测试),其中我使用我自己的/ 库调用来重载new/ delete运算符。但是在链接静态库时,我一直得到“未定义的引用”错误,即使我改变了和的顺序。但是一切都适用于连接这…...
英文购物网站建设/百度官网首页网址
异常: 除非另外还指定了 TOP 或 FOR XML,否则,ORDER BY 子句在视图、内联函数、派生表、子查询和公用表表达式中无效 原因: 参考:http://blog.csdn.net/wrm_nancy/article/details/17170115 因为这一步不返回表&#x…...
部门门户网站建设请示/关键词搜索广告
不啰嗦,直接复制工具类 /*** 在windows和linux系统下均可正常使用* Create by ysterfoxmail.com 2018/6/6/006 14:51*/ public class ProjectPath {//获取项目的根路径public final static String classPath;static {//获取的是classpath路径,适用于读取…...
丰台网站建设公司/灰色词快速排名方法
滑块区间组件 功能需求: 最小值为0,按照给定的最大值,生成区间范围;拖动滑块移动时,显示相应的范围区间,滑块条显示对应的状态;点击时,使最近的滑块移动到鼠标点击的位置。 默认效…...