現在讓我們來用PHP實現一個通過POP3協議收取信件的類吧,這個類中所用到的一些sock操作的函數,不另做特殊說明,請參考php的有關資料。通過這個實例,相信你也會和我一樣,感覺到PHP中對於sock操作的靈活、方便和功能的強大。
首先,我們來說明一下這個類中需要用到的一些內部成員變量:(這些變量應該都是對外封閉的,可是由於php對類的成員變量沒有private與publice之類的分別,只好就這麽直接定義了。這是PHP的一個令人遺憾的地方。)
1、成員變量說明
classpop3
{
var$hostname="";//POP主機名
var$port=110;//主機的POP3端口,一般是110號端口
var$timeout=5;//連接主機的最大超時時間
var$connection=0;//保存與主機的連接
var$state="DISCONNECTED";//保存當前的狀態
var$debug=0;//做為標識,是否在調試狀態,是的話,輸出調試信息
var$err_str='';//如果出錯,這裏保存錯誤信息
var$err_no;//如果出錯,這裏保存錯誤號碼
var$resp;//臨時保存服務器的響應信息
var$apop;//指示需要使用加密方式進行密碼驗證,一般服務器不需要
var$messages;//郵件數
var$size;//各郵件的總大小
var$mail_list;//一個數組,保存各個郵件的大小及其在郵件服務器上序號
var$head=array();//郵件頭的內容,數組
var$body=array();//郵件體的內容,數組;
2、當然,這其中的有些變量,僅通過這樣一個簡單的說明並不能完全了解如何使用,下面我就逐個來說明這個類實現中的一些主要方法:
Functionpop3($server="192.100.100.1",$port=110,$time_out=5)
{$this->hostname=$server;
$this->port=$port;
$this->timeout=$time_out;
returntrue;
}
熟悉面向對象編程的朋友一看就會知道,這是這個類的構造函數,在初始化這個類時,可以給出這幾個最基本的參數:pop3服務器的地址,端口號,及連接服務器時的最大超時時間。一般來說,只需要給出POP3服務器的地址就行了。
Functionopen()
{
if($this->hostname=="")
{$this->err_str="無效的主機名!!";
returnfalse;
}
if($this->debug)echo"正在打開$this->hostname,$this->port,&$err_no,&$err_str,$this->timeout<br/>";
if(!$this->connection=fsockopen($this->hostname,$this->port,&$err_no,&$err_str,$this->timeout))
{
$this->err_str="連接到POP服務器失敗,錯誤信息:".$err_str."錯誤號:".$err_no;
returnfalse;
}
else
{
$this->getresp();
if($this->debug)
$this->outdebug($this->resp);
if(substr($this->resp,0,3)!="+OK")
{$this->err_str="服務器返回無效的信息:".$this->resp."請檢查POP服務器是否正確";
returnfalse;
}
$this->state="AUTHORIZATION";
returntrue;
}
}
該方法不需要任何參數就可建立與POP3服務器的sock連接。該方法又用到了另一個類中的方法$this->getresp();下面是這個方法的聲明:
Functiongetresp()
{
for($this->resp="";;)
{
if(feof($this->connection))
returnfalse;
$this->resp.=fgets($this->connection,100);
$length=strlen($this->resp);
if($length>=2&&substr($this->resp,$length-2,2)=="\r\n")
{
$this->resp=strtok($this->resp,"\r\n");
returntrue;
}
}
}
這個方法取得服務器端的返回信息並進行簡單的處理:去掉最後的回車換行符,將返回信息保存在resp這個內部變量中。這個方法在後面的多個操作中都將用到。另外,還有個小方法也在後面的多個操作中用到:
Functionoutdebug($message)
{
echohtmlspecialchars($message)."<br/>\n";
}
它的作用就是把調試信息$message顯示出來,並把一些特殊字符進行轉換以及在行尾加上<br/>標簽,這樣是為了使其輸出的調試信息便於閱讀和分析。
建立起與服務器的sock連接之後,就要給服務器發送相關的命令了(請參見上面的與服務器對話的過程)從上面對POP對話的分析可以看到,每次都是發送一條命令,然後服務器給予一定的回應,如果命令的執行是對的,回應一般是以+OK開頭,後面是一些描述信息,所以,我們可以做一個通過發送命令的方法:
Functioncommand($command,$return_lenth=1,$return_code='+')
{
if($this->connection==0)
{
$this->err_str="沒有連接到任何服務器,請檢查網絡連接";
returnfalse;
}
if($this->debug)
$this->outdebug(">>>$command");
if(!fputs($this->connection,"$command\r\n"))
{
$this->err_str="無法發送命令".$command;
returnfalse;
}
else
{
$this->getresp();
if($this->debug)
$this->outdebug($this->resp);
if(substr($this->resp,0,$return_lenth)!=$return_code)
{
$this->err_str=$command."命令服務器返回無效:".$this->resp;
returnfalse;
}
else
returntrue;
}
}
這個方法可以接受三個參數:$command-->發送給服務器的命令;$return_lenth,$return_code,指定從服務器的返回中取多長的值做為命令返回的標識以及這個標識的正確值是什麽。對於一般的pop操作來說,如果服務器的返回第一個字符為"+",則可以認為命令是正確執行了。也可以用前面提到過的三個字符"+OK"做為判斷的標識。
下面介紹的幾個方法則可以按照前述收取信件的對話去理解,因為有關的內容已經在前面做了說明,因此下面的方法不做詳細的說明,請參考其中的註釋:
FunctionLogin($user,$password)//發送用戶名及密碼,登錄到服務器
{
if($this->state!="AUTHORIZATION")
{
$this->err_str="還沒有連接到服務器或狀態不對";
returnfalse;
}
if(!$this->apop)//服務器是否采用APOP用戶認證
{
if(!$this->command("USER$user",3,"+OK"))returnfalse;
if(!$this->command("PASS$password",3,"+OK"))returnfalse;
}
else
{
//echo$this->resp=strtok($this->resp,"\r\n");
if(!$this->command("APOP$user".md5($this->greeting.$password),3,"+OK"))returnfalse;
}
$this->state="TRANSACTION";//用戶認證通過,進入傳送模式
returntrue;
}
Functionstat()//對應著stat命令,取得總的郵件數與總的大小
{
if($this->state!="TRANSACTION")
{
$this->err_str="還沒有連接到服務器或沒有成功登錄";
returnfalse;
}
if(!$this->command("STAT",3,"+OK"))
returnfalse;
else
{
$this->resp=strtok($this->resp,"");
$this->messages=strtok("");//取得郵件總數
$this->size=strtok("");//取得總的字節大小
returntrue;
}
}
Functionlistmail($mess=null,$uni_id=null)//對應的是LIST命令,取得每個郵件的大小及序號。一般來說用到的是List命令,如果指定了$uni_id,則使用UIDL命令,返回的是每個郵件的標識符,事實上,這個標識符一般是沒有什麽用的。取得的各個郵件的大小返回到類的內部變量mail_list這個二維數組裏。
{
if($this->state!="TRANSACTION")
{
$this->err_str="還沒有連接到服務器或沒有成功登錄";
returnfalse;
}
if($uni_id)
$command="UIDL";
else
$command="LIST";
if($mess)
$command.=$mess;
if(!$this->command($command,3,"+OK"))
{
//echo$this->err_str;
returnfalse;
}
else
{
$i=0;
$this->mail_list=array();
$this->getresp();
while($this->resp!=".")
{$i++;
if($this->debug)
{
$this->outdebug($this->resp);
}
if($uni_id)
{
$this->mail_list[$i][num]=strtok($this->resp,"");
$this->mail_list[$i][size]=strtok("");
}
else
{
$this->mail_list[$i]["num"]=intval(strtok($this->resp,""));
$this->mail_list[$i]["size"]=intval(strtok(""));
}
$this->getresp();
}
returntrue;
}
}
functiongetmail($num=1,$line=-1)//取得郵件的內容,$num是郵件的序號,$line是指定共取得正文的多少行。有些時候,如郵件比較大而我們只想先查看郵件的主題時是必須指定行數的。默認值$line=-1,即取回所有的郵件內容,取得的內容存放到內部變量$head,$body兩個數組裏,數組裏的每一個元素對應的是郵件源代碼的一行。
{
?if($this->state!="TRANSACTION")
{
$this->err_str="不能收取信件,還沒有連接到服務器或沒有成功登錄";
returnfalse;
}
if($line<0)<br>$command="retr$num";<br>else<br>$command="top$num$line";<br><br>if(!$this->command("$command",3,"+OK"))
returnfalse;
else
{
$this->getresp();
$is_head=true;
while($this->resp!=".")//.號是郵件結束的標識
{
if($this->debug)
$this->outdebug($this->resp);
if(substr($this->resp,0,1)==".")
$this->resp=substr($this->resp,1,strlen($this->resp)-1);
if(trim($this->resp)=="")//郵件頭與正文部分的是一個空行
$is_head=false;
if($is_head)
$this->head[]=$this->resp;
else
$this->body[]=$this->resp;
$this->getresp();
}
returntrue;
}
}//endfunction
functiondele($num)//刪除指定序號的郵件,$num是服務器上的郵件序號
{
if($this->state!="TRANSACTION")
{
$this->err_str="不能刪除遠程信件,還沒有連接到服務器或沒有成功登錄";
returnfalse;
}
if(!$num)
{
$this->err_str="刪除的參數不對";
returnfalse;
}
if($this->command("DELE$num",3,"+OK"))
returntrue;
else
returnfalse;
}
通過以上幾個方法,我們已經可以實現郵件的查看、收取、刪除的操作,不過別忘了最後要退出,並關閉與服務器的連接,調用下面的這個方法:
FunctionClose()
{
if($this->connection!=0)
{
if($this->state=="TRANSACTION")
$this->command("QUIT",3,"+OK");
fclose($this->connection);
$this->connection=0;
$this->state="DISCONNECTED";
}
}