PHP发送邮件流程分析讲解
思韵闪耀
2013-03-19
0

网站应用中向客户发送邮件是常见的一个功能。SMTP协议貌似简单,而且资料繁多,但要彻底搞清楚客户端服务器之间的身份和关系处理,也不是件容易的事。
本文简明扼要对smtp交换过程中身份和条件进行说明,使初次接触smtp开发的能有个清晰的处理思路。错误之处请多指正。

先说说smtp的基本原理, 邮件客户端(outlook)和发送方smtp服务器之间,  发送方smtp服务器和接受方smtp服务器之间,走的都是smtp协议.

先说说我们最常见的情况:用outlook发邮件,从 jack@abc.com 发往 rose@xyz.com。 outlook里写好邮件后发送,会连接到帐号里登记的smtp服务器25端口,开始smtp会话:

   mail from: <jack@abc.com>
   rcpt to:<rose@xyz.com>

abc.com服务器发现邮件的mail from 是本域的(abc.com),这种情况下smtp服务器一般需要验证用户身份,验证通过提交邮件,邮件进入服务器的发送队列。 服务器投递邮件的进程或线程,扫描发送队列,取出邮件后分析要往哪发。 服务器发现邮件是发往 xyz.com,不是本域,先通过dns服务器查询 xyz.com域的mx记录,假设为 smtp.xyz.com,邮件投递进程连接 smtp.xyz.com 的25端口,开始smtp会话

   mail from: <jack@abc.com>
   rcpt to:<rose@xyz.com>

xyz.com服务器判断邮件是来自别的域,发往本域,所以不需要验证用户。不过有可能的情况是xyz.com服务器先检查一下发信服务器的ip地址,和mail from 里的域名对应的mx记录是否匹配,不匹配的拒收。如果没有符合什么拒收条件,那么xyz.com手下这封信,放入rose的邮件夹,rose可以通过pop3收取邮件。

如果 abc.com发送邮件失败的话,邮件进入发送失败队列,可能直接扔掉,或者再试着重发n次,如果都不成功,会通知发件人或别的什么,这要看服务器的处理。

现在再说说php发邮件,首先要搞清楚,要用什么身份发送邮件。第一种:把自己当作outlook之类的客户端,先连到发信服务器,提交邮件后让发件服务器往外发送。第二种:把自己当作发信服务器,直接通过smtp连接收件人的服务器发送邮件。

这里我不推荐第二种方式,原因是:如果你php所在服务器的域名,ip,mx记录没有严格设置好的话,一般收件服务器有很大的几率会拒收;发送邮件本身的传输过程时间无法控制,如果和收件服务器有很大的延时,会严重影响自己的web服务器的工作,另外也不具有失败重发等处理。

对于发送邮件,windows和unix和很大的差别。windows一般来说都是像outlook那样的方式,用smtp协议连到发件服务器发送邮件。unix有自己的传统方式,就是unix主机自带 smtp server,传统的就是sendmail。另外还有个“sendmail”的概念,这里说的是一个程序的名字,无论是sendmail,qmail,postfix等等,都提供这个命令程序,通过它,可以把邮件放入本地邮件发送队列,让sendmail,qmail,postfix之类的投递程序去投递发送。sendmail程序一般是从标准输入里读入邮件内容。php的mail()函数实际上会打开sendmail程序,通过标准输入把邮件内容传给它,由sendmail程序来发送,剩下的就不管了。

所以大家可以看到,php里关于mail() 这个函数的说明,如果是在windows上,一般是设置 php.ini里 SMTP 和 smtp_port 选项的值,通过类outlook客户端的方式发送,但是这种方式的致命弱点是不支持smtp验证,而现在邮件服务器基本都是需要smtp验证的。uxni下一般需要设置 sendmail_path ,来说明sendmail程序的路径。当然windows上也可以用sendmail程序发送。

如果是unix,最理想的情况是,你的php所在的web服务器本身也是个smtp server,并设置好了mx记录等等,或者你有个能发送邮件的sendmail程序(自己写,封装smtp发送)那么先配置好 php.ini 里的sendmail_path,通过mail()函数就直接发送了。

如果没有能用的sendmail程序,那么就要通过socket 用smtp协议发送了,php这方面有很多扩展的库,可以直接拿来用。建议连上一个服务器让他帮你发,不建议直接连收件服务器(原因前面说了)。这里还有就是不建议直接在网页上php里连接发送,推荐的方式是把邮件写到数据库或文件里,让另一个程序(php,perl,python 都可以写)扫描后通过sokcet连接发送。这样既给用户有了好的响应体验,也可以控制发送的过程。


【版权声明】
本站部分内容来源于互联网,本站不拥有所有权,不承担相关法律责任。如果发现本站有侵权的内容,欢迎发送邮件至masing@13sy.com 举报,并提供相关证据,一经查实,本站将立刻删除涉嫌侵权内容。

相关内容

将IIS设置克隆到新服务器...
将IIS设置克隆到新服务器的最佳方法是使用IIS配置文件。以下是详...
2025-06-06
搭建Git服务器及本机克隆...
Git是什么?Git是目前世界上最先进的分布式版本控制系统。SVN...
2025-03-17
ubuntu环境下搭建gi...
操作环境:服务器:Ubuntu 24.04.2 LTS+git 2...
2025-03-17
Windows serve...
Windows server2019安装Intel I219-V网...
2025-01-15
ubuntu配置DNS
ubuntu配置DNS在Ubuntu中配置DNS通常涉及编辑/et...
2024-11-17
SqlServer 数据库...
--查看数据库大小SELECT DB_NAME(database_...
2024-09-03

热门资讯

Exchange 2013/2... Exchange 2013/2016/2019邮件大小限制 可以在三个位置上配置Exchange上的...
SMTP 发送邮件错误码和解决... SMTP错误码/建议解决方法 错误总表 420 1. Timeout Communication P...
Exchange 2013 O... 最近在Exchange 2013环境中,遇到OWA/ECP无法访问的问题;用户是可以正常通过...
Roundcube 配置过各提... Fileinfo/mime_content_type configuration: OK Mimet...
彻底卸载Exchange Se... 一般来说,我们安装完Exchange Server 2016之后,可以通过正常卸载程序来进行卸载,但...
Exchange 2013批量... 一批量删除特定主题的邮件 1.1批量删除所有数据库中特定主题的邮件 1)群发了几封主题为backup...
快速查看已安装Exchange... Exchange 2010的版本号在Exchange管理控制台(EMC)显示的,好像不准确了。比如,...
在 Exchange 2016... 用户邮箱是与人员关联的 Exchange 邮箱,通常为每人一个邮箱。每个用户邮箱都有一个关联的 Ac...
exchange2013域名(... 需要对 接受的域 和 电子邮件地址策略进行配置: 首先外网域名要做解析和增加mx记录,这里就不具体说...
如何为AD 添加多个UPN后缀 今天我在群里看到有群友在问User principal name (UPN)多后缀的问题,我在这里就...