PHP中用Socket 发送电子邮件

  在作者所申请的几个PHP主页空间中,能够提供mail功能的差不多了,总是调用完成mail()函数之后就
  没什么下文了。但是电子邮件在网上生活中的作用越来太大作用我不想了,但是如果主页空间不支持mail()发送那么怎么办呢?我也想过通过socket来实现邮件发送
  ,但是无奈对用php进行socket编程不熟悉,再加上发送邮件要使用SMTP协议,还要阅读大量的英文了,
  所以一直也没有研究过。终于有一天我去读了一篇文章,关于用套接字编程发送邮件。我如获至宝般将其
  副本下来,并将其改造成了一个php可用的类,供大家使用。原来的文章只是一个简单的例子,而且还有
  一些我经过多次的实验、改造终于将其改造成了一个直接的使用socket,向指定的邮箱发送邮件的类,
  如果大家和前面关于发送MIME的文章结合起来,就可以实现在不支持mail()函数的网站上发送邮件了。因为
  发送邮件的过程需要时间,可能和mail()的处理机制不太完全一样,所以速度要慢一些,但是可以需要解决
  发送邮件功能的燃眉之急,同时你也可以学习用php进行socket编程。下面就将这类的实现原理向大家介绍,
  同时向大家讲解一些关于SMTP的基本知识
  。Socket编程
  向大家介绍申明,本人不是一个TCP/IP编程高手,故这里只是讲一下我的一点理解和经验。
  使用fsockopen函数打开一个Internet连接函数语法格式:
  int fsockopen(string hostname,int port,int[errno],string[errstr],int[timeout]);
  参数的意思我想不用讲了,这里由于要使用SMTP协议,所以端口号为25。在打开连接成功后,会返回
  一个socket句柄,使用它就可以象使用文件句柄一样的。可使用的操作有fputs(),fgets(),feof(),fclose()
  等。
  很简单地介绍就到这里吧。SMTP
  的基础
  基于TCP/IP的认知协议一般的命令格式都是通过请求/应答方式简单实现的,采用的都是文本信息,所以
  处理起来要容易一些。SMTP是邮件传输协议的意思是,它可以实现客户端向服务器发送邮件的功能。所以
  下面所讲的命令是指客户端向服务器发出请求指令,而响应则指返回服务器给客户端的信息。
  SMTP分成命令头和信息体两部分。命令头主要完成客户端与服务器的连接、验证。整个过程由多条
  命令每个命令发送到服务器后,由服务器给出响应信息,一般为3位等数字不同的
  服务器返回的响应码是遵守协议的,但是响应正文本则不一定。每个命令及响应的最后都有一个回车符,这样
  使用fputs()和fgets()就可以进行命令与响应的处理了。SMTP的命令及响应信息都是单行的。信息体蚴怯始?
  的正文部分,最后的结束行应以单独的"."作为结束行。
  客户端一些常用的SMTP指令为:
  HELO主机名:与服务器打招呼并告知客户端使用的机器名称,可以随便填写
  MAIL FROM:sender_id:告诉服务器发信人的地址
  RCPT TO:receive_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结束数据。
  致:chatme 263.net
  发件人:chatme 263.net
  主题:test
  发件人:chatme 263.net
  test
  。
  QUIT
  250 Ok:排队为C46411C5097E0
  这就是一些SMTP的简单知识。相关内容可以查阅RFC。RFC
  821定义了收/发电子邮件的相关指令。RFC
  822则制定了邮件内容的格式。RFC
  2045-2048制定了多媒体邮件内容的格式,
  RFC 1113,1422-1424写入讨论如何泄露电子邮件的保密性。send_mail
  类的实现
  下面现在开始介绍我所编写的发送邮件类。有了上面的基本知识了,就是实现了。
  类的成员
  变量$最后一条消息;//记录最后返回的响应信息
  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不能为NULL!");
  $this->smtp=$smtp;
  if(空($welcome))
  {
  $this->welcome=gethostbyaddr("localhost");
  }
  否则
  $this->welcome=$welcome;
  $this->debug=$debug;
  $this->lastmessage="";
  $this->lastact="";
  $这个->端口=“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否则
  10$m='>>';
  11 if(!ereg("\n$",$message))
  12$message.="<br>";
  分段阅读_第13章
  14 echo"<font color=#999999>${m}${message}</font>";
  15}
  16}
  --------------------------------------------------------------------------------------------
  这个函数用来显示调试信息。可以在$inout中指定是上传的指令还是返回的响应,如果为上传指令,则
  使用“out”;如果为返回的响应则使用“in”。
  第3行,判断是否要输出调试信息。
  第5行,判断是否为响应信息,如果是,则在第7行将信息的前面加上"<<"来区别信息;否则在第10行加上
  ">>"来区别上传指令。
  第11-12行,判断信息串最后是否为换行符,如不是则加上HTML换行标记。第13行将所以的换行符转成HTML
  的换行标记。
  第14行,输出整条信息,同时将信息颜色设置为灰色以示区别。
  执行一个命令
  --------------------------------------------------------------------------------
  1 function do_command($command,$code)
  2{
  3$this->lastact=$command;
  4$this->show_debug($this->lastact,“出去”);
  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返回false;
  11}
  12 else
  13返回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{
  3
  4//连接服务器
  5$this->lastact="connect";
  6
  7$this->show_debug("连接到SMTP服务器:".$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章
  分段阅读_第14章
  15
  16 if(!ereg("^220",$this->lastmessage))
  17{
  18 return false;
  19}
  20 else
  21{
  22$this->lastact="HELO".$this->欢迎。“\n”;
  23 if(!$this->do_command($this->lastact,"250"))
  24{
  25 fclose($this->fp);
  26返回假;
  27}
  28
  29$this->lastact="邮件发件人:$from".“\n”;
  30 if(!$this->do_command($this->lastact,"250"))
  31{
  32 fclose($this->fp);
  33返回假;
  34}
  35
  36$this->lastact="RCPT TO:$to"。“\n”;
  37 if(!$this->do_command($this->lastact,"250"))
  38{
  39 fclose($this->fp);
  40返回假;
  41}
  42
  43//发送正文
  44$this->lastact="DATA\n";
  45 if(!$this->do_command($this->lastact,"354"))
  46{
  47 fclose($this->fp);
  48返回假;
  49}
  50
  51//处理Subject头
  52$head="Subject:
  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章
  76 fputs($this->fp,$message);分段阅读_第77
  章79 if(!$this->do_command($this->lastact,"250"))80{81 fclose($this->fp);82返回假;83}84}85返回true;86}87 else 88{89$this->show_debug("连接失败!","in");90返回假;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包含“sendmail.class.php3”;2$email="您好,这是一封测试信!";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"发送成功!<br>";7}8 else 9{10 echo"发送失败!<br>";11}?>--------------------------------------------------------------------------------------------第1行,导入send_mail类。第3行,创建一个类的实例,且设置显示调示信息,如果不想显示,可以
  $sendmail=new send_mail("smtp.263.net");。
  第4行,发送邮件。
  很简单,不是吗?下面再给合以前的发送MIME邮件的例子,给出一个发送HTML附件的例子。
  --------------------------------------------------------------------------------
  <?php
  include"MIME.class.php3";
  //注,在发送MIME邮件一句中,这个类文件名为MIME.class,在此处我改成这样的
  $to='chatme 263.net';//改为收信人的邮箱
  $str="Newsletter for".date('M Y',time());
  //信息被我改少了
  $html_data='<html><head><title>'.$str。'</title></head>
  <body bgcolor="#ffffff">
  您好!这是一个测试!
  </body>
  </html>';
  //生成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."<br>";
  //包含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);
  ?>