0

0

用Socket发送电子邮件(php)

高洛峰

高洛峰

发布时间:2016-12-02 10:12:15

|

1306人浏览过

|

来源于php中文网

原创

socket编程介绍 
  向大家申明,本人不是一个tcp/ip编程专家,故在此只是讲出了我的一点理解和体会。

  使用fsockopen函数打开一个Internet连接,函数语法格式:

int fsockopen(string hostname, int port, int [errno], string [errstr], int [timeout]);

  参数的意思我想不用讲了,这里由于要使用SMTP协议,所以端口号为25。在打开连接成功后,会返回一 
个socket句柄,使用它就可以象使用文件句柄一样的。可使用的操作有fputs(),fgets(),feof(),fclose() 
等。

  很简单地介绍就到这里吧。

立即学习PHP免费学习笔记(深入)”;

SMTP的基础 
  基于TCP/IP的因特网协议一般的命令格式都是通过请求/ 应答方式实现的,采用的都是文本信息,所以 
处理起来要容易一些。SMTP是简单邮件传输协议的简称,它可以实现客户端向服务器发送邮件的功能。所以 
下面所讲的命令是指客户端向服务器发出请求指令,而响应则是指服务器返回给客户端的信息。

  SMTP分为命令头和信息体两部分。命令头主要完成客户端与服务器的连接,验证等。整个过程由多条命 
令组成。每个命令发到服务器后,由服务器给出响应信息,一般为3 位数字的响应码和响应文本。不同的服 
务器返回的响应码是遵守协议的,但是响应正文本则不必。每个命令及响应的最后都有一个回车符,这样使 
用fputs()和fgets()就可以进行命令与响应的处理了。SMTP的命令及响应信息都是单行的。信息体则是邮件 
的正文部分,最后的结束行应以单独的"."作为结束行。

  客户端一些常用的SMTP指令为:

HELO hostname: 与服务器打招呼并告知客户端使用的机器名字,可以随便填写 
MAIL FROM: sender_id : 告诉服务器发信人的地址 
RCPT TO: receiver_id : 告诉服务器收信人的地址 
DATA : 下面开始传输信件内容,且最后要以只含有.的特殊行结束 
RESET: 取消刚才的指令,从新开始 
VERIFY userid: 校验帐号是否存在(此指令为可选指令,服务器可能不支持) 
QUIT : 退出连接,结束 
  服务器返回的响应信息为(格式为:响应码+空格+解释):

220 服务就绪(在socket连接成功时,会返回此信息) 
221 正在处理 
250 请求邮件动作正确,完成(HELO,MAIL FROM,RCPT TO,QUIT指令执行成功会返回此信息) 
354 开始发送数据,结束以 .(DATA指令执行成功会返回此信息,客户端应发送信息) 
500 语法错误,命令不能识别 
550 命令不能执行,邮箱无效 
552 中断处理:用户超出文件空间 
  下面给出一个简单的命令头(这是在打开socket之后做的),是我向stmp.263.net发邮件的测试结果:

HELO limodou 
250 smtp.263.net 
MAIL FROM: chatme@263.net 
250 Ok 
RCPT TO: chatme@263.net 
250 Ok 
DATA 
354 End data with . 
To: chatme@263.net 
From: chatme@263.net 
Subject: test 
From: chatme@263.net 
test 

QUIT 
250 Ok: queued as C46411C5097E0

  这就是一些SMTP的简单知识。相关内容可以查阅RFC。

RFC 821定义了收/发电子邮件的相关指令。 
RFC 822则制定了邮件?热莸母袷健 
RFC 2045-2048制定了多媒体邮件?热莸母袷剑 
RFC 1113, 1422-1424则是讨论如何增进电子邮件的保密性。

send_mail类的实现 
  现在开始介绍我所编写的发送邮件类。有了上面的预备知识了,下面就是实现了。

类的成员变量

var $lastmessage; //记录最后返回的响应信息 
var $lastact; //最后的动作,字符串形式 
var $welcome; //用在HELO后面,欢迎用户 
var $debug; //是否显示调试信息 
var $smtp; //smtp服务器 
var $port; //smtp端口号 
var $fp; //socket句柄

  其中,$lastmessage和$lastact用于记录最后一次响应信息及执行的命令,当出错时,用户可以使用它 
们。为了测试需要,我还定义了$debug变量,当其值为true时,会在运行过程中显示一些执行信息,否则无 
任何输出。$fp用于保存打开后的socket句柄。

类的构造


-------------------------------------------------------------------------------- 
function send_mail($smtp, $welcome="", $debug=false) 

if(empty($smtp)) die("SMTP cannt be NULL!"); 
$this->smtp=$smtp; 
if(empty($welcome)) 

$this->welcome=gethostbyaddr("localhost"); 

else 
$this->welcome=$welcome; 
$this->debug=$debug; 
$this->lastmessage=""; 
$this->lastact=""; 
$this->port="25"; 

-------------------------------------------------------------------------------- 
  这个构造函数主要完成一些初始值的判定及设置。$welcome用于HELO指令中,告诉服务器用户的名字。 
HELO指令要求为机器名,但是不用也可以。如果用户没有给出$welcome,则自动查找本地的机器名。

显示调试信息

-------------------------------------------------------------------------------- 
1 function show_debug($message, $inout) 
2 { 
3 if ($this->debug) 
4 { 
5 if($inout=="in") //响应信息 
6 { 
7 $m='8 } 
9 else 
10 $m='>> '; 
11 if(!ereg("\n$", $message)) 
12 $message .= "
"; 
13 $message=nl2br($message); 
14 echo "${m}${message}"; 
15 } 
16 } 
-------------------------------------------------------------------------------- 
  这个函数用来显示调试信息。可以在$inout中指定是上传的指令还是返回的响应,如果为上传指令,则 
使用"out";如果为返回的响应则使用"in"。

第3行,判断是否要输出调试信息。 
第5行,判断是否为响应信息,如果是,则在第7行将信息的前面加上"    ">> "来区别上传指令。 
第11-12行,判断信息串最后是否为换行符,如不是则加上HTML换行标记。第13行将所以的换行符转成HTML 
    的换行标记。 
第14行,输出整条信息,同时将信息颜色置为灰色以示区别。

执行一个命令


-------------------------------------------------------------------------------- 
1 function do_command($command, $code) 
2 { 
3 $this->lastact=$command; 
4 $this->show_debug($this->lastact, "out"); 
5 fputs ( $this->fp, $this->lastact ); 
6 $this->lastmessage = fgets ( $this->fp, 512 ); 
7 $this->show_debug($this->lastmessage, "in"); 
8 if(!ereg("^$code", $this->lastmessage)) 
9 { 
10 return false; 
11 } 
12 else 
13 return true; 
14 } 
-------------------------------------------------------------------------------- 
  在编写socket处理部分发现,一些命令的处理很相似,如HELO,MAIL FROM,RCPT TO,QUIT,DATA命令, 
都要求根据是否显示调试信息将相关内容显示出来,同时对于返回的响应码,如果是期望的,则应继续处理, 
如果不是期望的,则应中断出理。所以为了清晰与简化,专门对这些命令的处理编写了一个通用处理函数。 
函数的参数中$code为期望的响应码,如果响应码与之相同则表示处理成功,否则出错。

第3行,记录最后执行命令。 
第4行,将上传命令显示出来。 
第5行,则使用fputs真正向服务器传换指令。 
第6行,从服务器接收响应信息将放在最后响应消息变量中。 
第7行,将响应信息显示出来。 
第8行,判断响应信息是否期待的,如果是则第13行返回成功(true),否则在第10行返回失败(false)。

  这样,这个函数一方面完成指令及信息的发送显示功能,别一方面对返回的响应判断是否成功。

邮件发送处理

  下面是真正的秘密了,可要看仔细了。:)

-------------------------------------------------------------------------------- 
1 function send( $to,$from,$subject,$message) 
2 { 

4 //连接服务器 
5 $this->lastact="connect"; 

7 $this->show_debug("Connect to SMTP server : ".$this->smtp, "out"); 
8 $this->fp = fsockopen ( $this->smtp, $this->port ); 
9 if ( $this->fp ) 
10 { 
11 
12 set_socket_blocking( $this->fp, true ); 
13 $this->lastmessage=fgets($this->fp,512); 
14 $this->show_debug($this->lastmessage, "in"); 
15 
16 if (! ereg ( "^220", $this->lastmessage ) ) 
17 { 
18 return false; 
19 } 
20 else 
21 { 
22 $this->lastact="HELO " . $this->welcome . "\n"; 
23 if(!$this->do_command($this->lastact, "250")) 
24 { 
25 fclose($this->fp); 
26 return false; 
27 } 
28 
29 $this->lastact="MAIL FROM: $from" . "\n"; 
30 if(!$this->do_command($this->lastact, "250")) 
31 { 
32 fclose($this->fp); 
33 return false; 
34 } 
35 
36 $this->lastact="RCPT TO: $to" . "\n"; 
37 if(!$this->do_command($this->lastact, "250")) 
38 { 
39 fclose($this->fp); 
40 return false; 
41 } 
42 
43 //发送正文 
44 $this->lastact="DATA\n"; 
45 if(!$this->do_command($this->lastact, "354")) 
46 { 
47 fclose($this->fp); 
48 return false; 
49 } 
50 
51 //处理Subject头 
52 $head="Subject: $subject\n"; 
53 if(!empty($subject) && !ereg($head, $message)) 
54 { 
55 $message = $head.$message; 
56 } 
57 
58 //处理From头 
59 $head="From: $from\n"; 
60 if(!empty($from) && !ereg($head, $message)) 
61 { 
62 $message = $head.$message; 
63 } 
64 
65 //处理To头 
66 $head="To: $to\n"; 
67 if(!empty($to) && !ereg($head, $message)) 
68 { 
69 $message = $head.$message; 
70 } 
71 
72 //加上结束串 
73 if(!ereg("\n\.\n", $message)) 
74 $message .= "\n.\n"; 
75 $this->show_debug($message, "out"); 
76 fputs($this->fp, $message); 
77 
78 $this->lastact="QUIT\n"; 
79 if(!$this->do_command($this->lastact, "250")) 
80 { 
81 fclose($this->fp); 
82 return false; 
83 } 
84 } 
85 return true; 
86 } 
87 else 
88 { 
89 $this->show_debug("Connect failed!", "in"); 
90 return false; 
91 } 
92 } 
-------------------------------------------------------------------------------- 
  有些意思很清楚的我就不说了。

  这个函数一共有四个参数,分别是$to表示收信人,$from表示发信人,$subject表求邮件主题和$message 
表示邮件体。如果处理成功则返回true,失败则返回false。

第8行,连接邮件服务器,如果成功响应码应为220。 
第12行,设置阻塞模式,表示信息必须返回才能继续。详细说明看手册吧。 
第16行,判断响应码是否为220,如果是,则继续处理,否则出错返回。 
第22-27行,处理HELO指令,期望响应码为250。 
第29-34行,处理MAIL FROM指令,期望响应码为250。 
第36-41行,处理RCPT TO指令,期望响应码为250。 
第44-49行,处理DATA指令,期望响应码为354。 
第51-76行,生成邮件体,并发送。 
第52-56行,如果$subject不为空,则查找邮件体中是否有主题部分,如果没有,则加上主题部分。 
第59-63行,如果$from不为空,则查找邮件体中是否有发信人部分,如果没有,则加上发信人部分。 
第66-70行,如果$to不为空,则查找邮件体中是否有收信人部分,如果没有,则加上收信人部分。 
第73-74行,查找邮件体是否有了结束行,如果没有则加上邮件体的结束行(以"."作为单独的一行的特殊行)。 
第76行,发送邮件体。 
第78-83行,执行QUIT结否与服务器的连接,期望响应码为250。 
第85行,返回处理成功标志(true)。 
第81-91行,与服务器连接失败的处理。

  以上为整个send_mail类的实现,应该不是很难的。下面给出一个实例。

邮件发送实例 
  先给出一个最简单的实例: 
-------------------------------------------------------------------------------- 
 
1 include "sendmail.class.php3"; 
2 $email="Hello, this is a test letter!"; 
3 $sendmail=new send_mail("smtp.263.net", "limodou", true); //显示调示信息 
4 if($sendmail->send("chatme@263.net", "chatme@263.net", "test", $email)) 
5 { 
6 echo "发送成功!
"; 
7 } 
8 else 
9 { 
10 echo "发送失败!
"; 
11 } 
?> 
-------------------------------------------------------------------------------- 
第1行,装入send_mail类。 
第3行,创建一个类的实例,且设置显示调示信息,如果不想显示,可以 
    $sendmail=new send_mail("smtp.263.net");。 
第4行,发送邮件。


很简单,不是吗?下面再给合以前的发送MIME邮件的例子,给出一个发送HTML附件的例子。

-------------------------------------------------------------------------------- 

include "MIME.class.php3"; 
//注,在发送MIME邮件一文中,这个类文件名为MIME.class,在此处我改成这样的

$to = 'chatme@263.net'; //改为收信人的邮箱 
$str = "Newsletter for ".date('M Y', time());

//信息被我改少了 
$html_data = '

'. $str. ' 
 
Hello! This is a test! 
 
';

//生成MIME类实例 
$mime = new MIME_mail("chatme@263.net", $to, $str);

//添加HTML附件 
$mime->attach($html_data, "", HTML, BASE64);

//注释掉,采用我的发送邮件处理 
//$mime->send_mail();

//生成邮件 
$mime->gen_email();

//显示邮件信息 
//echo $mime->email."
";

//包含sendmail文件 
include "sendmail.class.php3";

//创建实例 
$sendmail=new send_mail("smtp.263.net", "limodou", true);

//发送邮件 
$sendmail->send("chatme@263.net", "chatme@263.net", $str, $mime->email);

?>

PHP速学教程(入门到精通)
PHP速学教程(入门到精通)

PHP怎么学习?PHP怎么入门?PHP在哪学?PHP怎么学才快?不用担心,这里为大家提供了PHP速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
全国统一发票查询平台入口合集
全国统一发票查询平台入口合集

本专题整合了全国统一发票查询入口地址合集,阅读专题下面的文章了解更多详细入口。

19

2026.02.03

短剧入口地址汇总
短剧入口地址汇总

本专题整合了短剧app推荐平台,阅读专题下面的文章了解更多详细入口。

27

2026.02.03

植物大战僵尸版本入口地址汇总
植物大战僵尸版本入口地址汇总

本专题整合了植物大战僵尸版本入口地址汇总,前往文章中寻找想要的答案。

15

2026.02.03

c语言中/相关合集
c语言中/相关合集

本专题整合了c语言中/的用法、含义解释。阅读专题下面的文章了解更多详细内容。

3

2026.02.03

漫蛙漫画网页版入口与正版在线阅读 漫蛙MANWA官网访问专题
漫蛙漫画网页版入口与正版在线阅读 漫蛙MANWA官网访问专题

本专题围绕漫蛙漫画(Manwa / Manwa2)官网网页版入口进行整理,涵盖漫蛙漫画官方主页访问方式、网页版在线阅读入口、台版正版漫画浏览说明及基础使用指引,帮助用户快速进入漫蛙漫画官网,稳定在线阅读正版漫画内容,避免误入非官方页面。

13

2026.02.03

Yandex官网入口与俄罗斯搜索引擎访问指南 Yandex中文登录与网页版入口
Yandex官网入口与俄罗斯搜索引擎访问指南 Yandex中文登录与网页版入口

本专题汇总了俄罗斯知名搜索引擎 Yandex 的官网入口、免登录访问地址、中文登录方法与网页版使用指南,帮助用户稳定访问 Yandex 官网,并提供一站式入口汇总。无论是登录入口还是在线搜索,用户都能快速获取最新稳定的访问链接与使用指南。

114

2026.02.03

Java 设计模式与重构实践
Java 设计模式与重构实践

本专题专注讲解 Java 中常用的设计模式,包括单例模式、工厂模式、观察者模式、策略模式等,并结合代码重构实践,帮助学习者掌握 如何运用设计模式优化代码结构,提高代码的可读性、可维护性和扩展性。通过具体示例,展示设计模式如何解决实际开发中的复杂问题。

3

2026.02.03

C# 并发与异步编程
C# 并发与异步编程

本专题系统讲解 C# 异步编程与并发控制,重点介绍 async 和 await 关键字、Task 类、线程池管理、并发数据结构、死锁与线程安全问题。通过多个实战项目,帮助学习者掌握 如何在 C# 中编写高效的异步代码,提升应用的并发性能与响应速度。

2

2026.02.03

Python 强化学习与深度Q网络(DQN)
Python 强化学习与深度Q网络(DQN)

本专题深入讲解 Python 在强化学习(Reinforcement Learning)中的应用,重点介绍 深度Q网络(DQN) 及其实现方法,涵盖 Q-learning 算法、深度学习与神经网络的结合、环境模拟与奖励机制设计、探索与利用的平衡等。通过构建一个简单的游戏AI,帮助学习者掌握 如何使用 Python 训练智能体在动态环境中作出决策。

3

2026.02.03

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
进程与SOCKET
进程与SOCKET

共6课时 | 0.4万人学习

golang socket 编程
golang socket 编程

共2课时 | 0.1万人学习

swoole入门物联网开发与实战
swoole入门物联网开发与实战

共15课时 | 1.3万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号