PHP调用Python快速发送高并发邮件
off999 2024-12-12 14:24 13 浏览 0 评论
1 简介
在PHP中发送邮件,通常都是封装一个php的smtp邮件类来发送邮件。但是PHP底层的socket编程相对于python来说效率是非常低的。CleverCode同时写过用python写的爬虫抓取网页,和用php写的爬虫抓取网页。发现虽然用了php的curl抓取网页,但是涉及到超时,多线程同时抓取等等。不得不说python在网络编程的效率要比PHP好的多。
PHP在发送邮件时候,自己写的smtp类,发送的效率和速度都比较低。特别是并发发送大量带有附件报表的邮件的时候。php的效率很低。建议可以使用php调用python的方式来发送邮件。
2 程序
2.1 python程序
php的程序和python的文件必须是相同的编码。如都是gbk编号,或者同时utf-8编码,否则容易出现乱码。python发送邮件主要使用了email模块。这里python文件和php文件都是gbk编码,发送的邮件标题内容与正文内容也是gbk编码。
#!/usr/bin/python
# -*- coding:gbk -*-
"""
邮件发送类
"""
# mail.py
#
# Copyright (c) 2014 by http://blog.csdn.net/CleverCode
#
# modification history:
# --------------------
# 2014/8/15, by CleverCode, Create
import threading
import time
import random
from email.MIMEText import MIMEText
from email.MIMEMultipart import MIMEMultipart
from email.MIMEBase import MIMEBase
from email import Utils, Encoders
import mimetypes
import sys
import smtplib
import socket
import getopt
import os
class SendMail:
def __init__(self,smtpServer,username,password):
"""
smtpServer:smtp服务器,
username:登录名,
password:登录密码
"""
self.smtpServer = smtpServer
self.username = username
self.password = password
def genMsgInfo(self,fromAddress,toAddress,subject,content,fileList,\
subtype = 'plain',charset = 'gb2312'):
"""
组合消息发送包
fromAddress:发件人,
toAddress:收件人,
subject:标题,
content:正文,
fileList:附件,
subtype:plain或者html
charset:编码
"""
msg = MIMEMultipart()
msg['From'] = fromAddress
msg['To'] = toAddress
msg['Date'] = Utils.formatdate(localtime=1)
msg['Message-ID'] = Utils.make_msgid()
#标题
if subject:
msg['Subject'] = subject
#内容
if content:
body = MIMEText(content,subtype,charset)
msg.attach(body)
#附件
if fileList:
listArr = fileList.split(',')
for item in listArr:
#文件是否存在
if os.path.isfile(item) == False:
continue
att = MIMEText(open(item).read(), 'base64', 'gb2312')
att["Content-Type"] = 'application/octet-stream'
#这里的filename邮件中显示什么名字
filename = os.path.basename(item)
att["Content-Disposition"] = 'attachment; filename=' + filename
msg.attach(att)
return msg.as_string()
def send(self,fromAddress,toAddress,subject = None,content = None,fileList = None,\
subtype = 'plain',charset = 'gb2312'):
"""
邮件发送函数
fromAddress:发件人,
toAddress:收件人,
subject:标题
content:正文
fileList:附件列表
subtype:plain或者html
charset:编码
"""
try:
server = smtplib.SMTP(self.smtpServer)
#登录
try:
server.login(self.username,self.password)
except smtplib.SMTPException,e:
return "ERROR:Authentication failed:",e
#发送邮件
server.sendmail(fromAddress,toAddress.split(',') \
,self.genMsgInfo(fromAddress,toAddress,subject,content,fileList,subtype,charset))
#退出
server.quit()
except (socket.gaierror,socket.error,socket.herror,smtplib.SMTPException),e:
return "ERROR:Your mail send failed!",e
return 'OK'
def usage():
"""
使用帮助
"""
print """Useage:%s [-h] -s <smtpServer> -u <username> -p <password> -f <fromAddress> -t <toAddress> [-S <subject> -c
<content> -F <fileList>]
Mandatory arguments to long options are mandatory for short options too.
-s, --smtpServer= smpt.xxx.com.
-u, --username= Login SMTP server username.
-p, --password= Login SMTP server password.
-f, --fromAddress= Sets the name of the "from" person (i.e., the envelope sender of the mail).
-t, --toAddress= Addressee's address. -t "test@test.com,test1@test.com".
-S, --subject= Mail subject.
-c, --content= Mail message.-c "content, ......."
-F, --fileList= Attachment file name.
-h, --help Help documen.
""" %sys.argv[0]
def start():
"""
"""
try:
options,args = getopt.getopt(sys.argv[1:],"hs:u:p:f:t:S:c:F:","--help --smtpServer= --username= --password= --fromAddress= --toAddress= --subject= --content= --fileList=",)
except getopt.GetoptError:
usage()
sys.exit(2)
return
smtpServer = None
username = None
password = None
fromAddress = None
toAddress = None
subject = None
content = None
fileList = None
#获取参数
for name,value in options:
if name in ("-h","--help"):
usage()
return
if name in ("-s","--smtpServer"):
smtpServer = value
if name in ("-u","--username"):
username = value
if name in ("-p","--password"):
password = value
if name in ("-f","--fromAddress"):
fromAddress = value
if name in ("-t","--toAddress"):
toAddress = value
if name in ("-S","--subject"):
subject = value
if name in ("-c","--content"):
content = value
if name in ("-F","--fileList"):
fileList = value
if smtpServer == None or username == None or password == None:
print 'smtpServer or username or password can not be empty!'
sys.exit(3)
mail = SendMail(smtpServer,username,password)
ret = mail.send(fromAddress,toAddress,subject,content,fileList)
if ret != 'OK':
print ret
sys.exit(4)
print 'OK'
return 'OK'
if __name__ == '__main__':
start()
2.2 python程序使用帮助
输入以下命令,可以输出这个程序的使用帮助
# python mail.py --help
2.3 php程序
这个程序主要是php拼接命令字符串,调用python程序。注意:用程序发送邮件,需要到邮件服务商,开通stmp服务功能。如qq就需要开通smtp功能后,才能用程序发送邮件。开通如下图。
php调用程序如下:
<?php
/**
* SendMail.php
*
* 发送邮件类
*
* Copyright (c) 2015 by http://blog.csdn.net/CleverCode
*
* modification history:
* --------------------
* 2015/5/18, by CleverCode, Create
*
*/
class SendMail{
/**
* 发送邮件方法
*
* @param string $fromAddress 发件人,'clevercode@qq.com' 或者修改发件人名 'CleverCode<clevercode@qq.com>'
* @param string $toAddress 收件人,多个收件人逗号分隔,'test1@qq.com,test2@qq.com,test3@qq.com....', 或者 'test1<test1@qq.com>,test2<test2@qq.com>,....'
* @param string $subject 标题
* @param string $content 正文
* @param string $fileList 附件,附件必须是绝对路径,多个附件逗号分隔。'/data/test1.txt,/data/test2.tar.gz,...'
* @return string 成功返回'OK',失败返回错误信息
*/
public static function send($fromAddress, $toAddress, $subject = NULL, $content = NULL, $fileList = NULL){
if (strlen($fromAddress) < 1 || strlen($toAddress) < 1) {
return '$fromAddress or $toAddress can not be empty!';
}
// smtp服务器
$smtpServer = 'smtp.qq.com';
// 登录用户
$username = 'clevercode@qq.com';
// 登录密码
$password = '123456';
// 拼接命令字符串,实际是调用了/home/CleverCode/mail.py
$cmd = "LANG=C && /usr/bin/python /home/CleverCode/mail.py";
$cmd .= " -s '$smtpServer'";
$cmd .= " -u '$username'";
$cmd .= " -p '$password'";
$cmd .= " -f '$fromAddress'";
$cmd .= " -t '$toAddress'";
if (isset($subject) && $subject != NULL) {
$cmd .= " -S '$subject'";
}
if (isset($content) && $content != NULL) {
$cmd .= " -c '$content'";
}
if (isset($fileList) && $fileList != NULL) {
$cmd .= " -F '$fileList'";
}
// 执行命令
exec($cmd, $out, $status);
if ($status == 0) {
return 'OK';
} else {
return "Error,Send Mail,$fromAddress,$toAddress,$subject,$content,$fileList ";
}
return 'OK';
}
}
2.3 使用样例
压缩excel成附件,发送邮件。
<?php
/**
* test.php
*
* 压缩excel成附件,发送邮件
*
* Copyright (c) 2015 http://blog.csdn.net/CleverCode
*
* modification history:
* --------------------
* 2015/5/14, by CleverCode, Create
*
*/
include_once ('SendMail.php');
/*
* 客户端类
* 让客户端和业务逻辑尽可能的分离,降低页面逻辑和业务逻辑算法的耦合,
* 使业务逻辑的算法更具有可移植性
*/
class Client{
public function main(){
// 发送者
$fromAddress = 'CleverCode<clevercode@qq.com>';
// 接收者
$toAddress = 'all@qq.com';
// 标题
$subject = '这里是标题!';
// 正文
$content = "您好:\r\n";
$content .= " 这里是正文\r\n ";
// excel路径
$filePath = dirname(__FILE__) . '/excel';
$sdate = date('Y-m-d');
$PreName = 'CleverCode_' . $sdate;
// 文件名
$fileName = $filePath . '/' . $PreName . '.xls';
// 压缩excel文件
$cmd = "cd $filePath && zip $PreName.zip $PreName.xls";
exec($cmd, $out, $status);
$fileList = $filePath . '/' . $PreName . '.zip';
// 发送邮件(附件为压缩后的文件)
$ret = SendMail::send($fromAddress, $toAddress, $subject, $content, $fileList);
if ($ret != 'OK') {
return $ret;
}
return 'OK';
}
}
/**
* 程序入口
*/
function start(){
$client = new Client();
$client->main();
}
start();
?>
2.4 程序源码下载
http://download.csdn.net/detail/clevercode/8711809
版权声明:
1)原创作品,出自"CleverCode的博客",转载时请务必注明以下原创地址,否则追究版权法律责任。
2)原创地址:http://blog.csdn.net/clevercode/article/details/45815453(转载务必注明该地址)。
3)欢迎大家关注我博客更多的精彩内容:http://blog.csdn.net/CleverCode。
相关推荐
- 独家 | 5 个Python高级特性让你在不知不觉中成为Python高手
-
你已经使用Python编程了一段时间,编写脚本并解决各种问题。是你的水平出色吗?你可能只是在不知不觉中利用了Python的高级特性。从闭包(closure)到上下文管理器(contextmana...
- Python装饰器
-
Python装饰器是一种用于修改函数或类的行为的特殊语法。它们允许在不修改原始代码的情况下,通过将函数或类作为参数传递给另一个函数来添加额外的功能。装饰器本质上是一个函数,它接受一个函数作为参数,并返...
- 中高阶Python常规用法--上下文管理器
-
Python以简单性和通用性著称,是一种深受全球开发人员喜爱的编程语言。它提供了大量的特性和功能,使编码成为一种愉快的体验。在这些功能中,一个经常被新手忽视的强大工具是上下文管理器。上下文管理器是高...
- Python小案例67- 装饰器
-
Python装饰器是一种用于修改函数或类的行为的特殊语法。它们允许在不修改原始代码的情况下,通过将函数或类作为参数传递给另一个函数来添加额外的功能。装饰器本质上是一个函数,它接受一个函数作为参数,并返...
- python常用的语法糖
-
概念Python的语法糖(SyntacticSugar)是指那些让代码更简洁、更易读的语法特性,它们本质上并不会增加新功能,但能让开发者更高效地编写代码。推导式写法推导式是Python最经典的...
- python - 常用的装饰器 decorator 有哪些?
-
python编程中使用装饰器(decorator)工具,可以使代码更简洁清晰,提高代码的重用性,还可以为代码维护提供方便。对于python初学者来说,根据装饰器(decorator)的字面意思并不...
- python数据缓存怎么搞 ?推荐一个三方包供你参考,非常简单好用。
-
1.数据缓存说明数据缓存可以说也是项目开发中比不可少的一个工具,像我们测试的系统中,你都会见到像Redis一样的数据缓存库。使用缓存数据库的好处不言而喻,那就是效率高,简单数据直接放在缓存中...
- 用于时间序列数据的Graphite监视工具
-
结合第三方工具,Graphite为IT性能监控提供了许多好处。本文介绍其核心组件,包括Carbon、Whisper以及安装的基本准则。Graphite监视工具可实时或按需,大规模地绘制来自多个来源的时...
- Python3+pygame实现的坦克大战
-
一、显示效果二、代码1.说明几乎所有pygame游戏,基本都遵循一定的开发流程,大体如下:初始化pygame创建窗口while循环检测以及处理事件(鼠标点击、按键等)更新UI界面2.代码创建一个m...
- Python之鸭子类型:一次搞懂with与上下文装饰器
-
引言在鸭子类型的理念的基础之上,从关注类型,转变到关注特性和行为。结合Python中的魔法函数的体系,我们可以将自定义的类型,像内置类型一样被使用。今天这篇文章中,接着该话题,继续聊一下with语法块...
- Python必会的50个代码操作
-
学习Python时,掌握一些常用的程序操作非常重要。以下是50个Python必会的程序操作,主要包括基础语法、数据结构、函数和文件操作等。1.HelloWorldprint("Hello,...
- 一文掌握Python 中的同步和异步
-
同步代码(Sync)同步就像在一个流水线上工作,每个任务都等待前一个任务完成。示例:机器A切割钢板→完成后,机器B钻孔→完成后,机器C上色。在Python中,同步代码看起来像这样:im...
- python 标注模块timeit: 测试函数的运行时间
-
在Python中,可以使用内置的timeit模块来测试函数的运行时间。timeit模块提供了一个简单的接口来测量小段代码的执行时间。以下是使用timeit测试函数运行时间的一般步骤:导入...
- Python带你找回童年的万花尺
-
还记得小时候的万花尺吧?这么画:一点也不费脑筋,就可以出来这么多丰富多彩的复杂几何图形。具体而言,可以用万花尺玩具(如图2-1所示)来绘制数学曲线。这种玩具由两个不同尺寸的塑料齿轮组成,一大一小。小的...
- Python 时间模块深度解析:从基础到高级的全面指南
-
直接上干货一、时间模块核心类介绍序号类名说明1datetime.datetime表示一个具体的日期和时间,结合了日期和时间的信息。2datetime.date表示一个具体的日期。3datetime.t...
你 发表评论:
欢迎- 一周热门
- 最近发表
- 标签列表
-
- python计时 (54)
- python安装路径 (54)
- python类型转换 (75)
- python进度条 (54)
- python的for循环 (56)
- python串口编程 (60)
- python写入txt (51)
- python读取文件夹下所有文件 (59)
- java调用python脚本 (56)
- python操作mysql数据库 (66)
- python字典增加键值对 (53)
- python获取列表的长度 (64)
- python接口 (63)
- python调用函数 (57)
- python qt (52)
- python人脸识别 (54)
- python斐波那契数列 (51)
- python多态 (60)
- python命令行参数 (53)
- python匿名函数 (59)
- python打印九九乘法表 (65)
- centos7安装python (53)
- python赋值 (62)
- python异常 (69)
- python元祖 (57)