簡介:這是在PHP中用Socket寄送電子郵件的詳細頁面,介紹了和php,有關的知識、技巧、經驗,和一些php源碼等。
class='pingjiaF' frameborder='0' src='http://biancheng.dnbcw.info/pingjia.php?id=330116' scrolling='no'>
在作者所申請的幾個PHP 首頁空間中,能夠提供mail功能的實在不多,總是調用完mail()函數之後就毫無下文了。但是電子郵件在網上生活中的作用越來越大。想一想網蟲上網不收郵件能叫真正的網蟲嗎?郵件的作用我不想再說了,但是如果首頁空間不支援mail()發送那麼怎麼辦呢?我也想過通過socket來實現郵件發送,但無奈對用php 進行socket編程不熟悉,再加上發送郵件要用到SMTP協議,又要讀不少的英文了,所以一直也沒有去研究過。終於有一天我發現了一篇文章,關於用socket編程發送郵件。我如獲至寶般將其拷貝下來,並且將其改造成了一個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 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行將資訊的前面加上"<< "來區別資訊;否則在第10行加上">> "來區別上傳指令。
第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 {
3
4 //串連伺服器
5 $this->lastact="connect";
6
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 = '
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);
?>
-------------------------------------------------------------
注釋寫的很清楚,就不再做更多的解釋了。如果實際應用中,請將send_mail建構函式中的debug設為false或不寫即可。
“在PHP中用Socket寄送電子郵件”的更多相關文章 》
愛J2EE關注Java邁克爾傑克遜視頻站JSON線上工具
http://biancheng.dnbcw.info/php/330116.html pageNo:12