雖然此答案的某些部分僅適用於 mail()
函數(shù)本身的使用,但其中許多故障排除步驟可以應(yīng)用於任何 PHP 郵件系統(tǒng)。 嗯>
您的腳本未發(fā)送電子郵件的原因有很多。除非有明顯的語法錯誤,否則很難診斷這些事情。如果沒有,您需要仔細檢查下面的清單,以查找您可能遇到的任何潛在陷阱。
錯誤報告對於根除程式碼中的錯誤以及 PHP 遇到的一般錯誤至關(guān)重要。需要啟用錯誤報告才能接收這些錯誤。將以下程式碼放在 PHP 檔案的頂部(或主設(shè)定檔中)將啟用錯誤報告。
error_reporting(-1); ini_set('display_errors', 'On'); set_error_handler("var_dump");
請參閱如何獲得有用的錯誤PHP 中的消息? — 此答案以了解更多詳細資訊。
mail()
函數(shù)這可能看起來很愚蠢,但一個常見的錯誤是忘記在程式碼中實際放置 mail()
函數(shù)。確保它在那裡並且沒有被註釋掉。
mail()
函數(shù)郵件函數(shù)採用三個必要參數(shù),以及可選的第四個和第五個參數(shù)。如果您對 mail()
的呼叫沒有至少三個參數(shù),它將失敗。
如果您對 mail()
的呼叫沒有以正確順序提供正確的參數(shù),它也會失敗。
您的網(wǎng)站伺服器應(yīng)該記錄透過它發(fā)送電子郵件的所有嘗試。這些日誌的位置會有所不同(您可能需要詢問伺服器管理員它們的位置),但通常可以在使用者根目錄的 logs
下找到它們。裡面將是伺服器報告的與您嘗試發(fā)送電子郵件相關(guān)的錯誤訊息(如果有)。
連接埠阻止是大多數(shù)開發(fā)人員在整合程式碼以使用 SMTP 發(fā)送電子郵件時面臨的一個非常常見的問題。並且,這可以在伺服器郵件日誌中輕鬆追蹤(郵件日誌伺服器的位置可能因伺服器而異,如上所述)。如果您位於共用主機伺服器上,則預(yù)設(shè)情況下連接埠 25 和 587 仍處於封鎖狀態(tài)。此封鎖是由您的託管提供者故意完成的。即使對於某些專用伺服器也是如此。當(dāng)這些連接埠被封鎖時,請嘗試使用連接埠 2525 進行連線。如果您發(fā)現(xiàn)該連接埠也被阻止,那麼唯一的解決方案就是聯(lián)絡(luò)您的託管提供者以解除這些連接埠的阻止。
大多數(shù)託管提供者都會阻止這些電子郵件端口,以保護其網(wǎng)路免於發(fā)送任何垃圾郵件。
使用連接埠 25 或 587 進行普通/TLS 連接,使用連接埠 465 進行 SSL 連接。對於大多數(shù)用戶,建議使用連接埠 587 以避免某些託管提供者設(shè)定的速率限制。
當(dāng)錯誤抑制運算子 @
被加入到 PHP 中的表達式之前,該表達式可能產(chǎn)生的任何錯誤訊息都會被忽略。在某些情況下,需要使用此運算符,但發(fā)送郵件不是其中之一。
如果您的程式碼包含@mail(...)
,那麼您可能隱藏了有助於您偵錯的重要錯誤訊息。刪除@
並查看是否報告任何錯誤。
僅當(dāng)您使用error_get_last()
緊接著具體失敗。
mail()
傳回值mail()
函數(shù):
這一點值得注意,因為:
FALSE
回傳值,您就知道錯誤在於您的伺服器接受您的郵件。這可能不是編碼問題,而是伺服器配置問題。您需要與系統(tǒng)管理員聯(lián)繫以找出發(fā)生這種情況的原因。 TRUE
傳回值,並不表示您的電子郵件一定會發(fā)送。這僅表示電子郵件已透過 PHP 成功傳送到伺服器上對應(yīng)的處理程序。還有更多 PHP 無法控制的故障點可能導(dǎo)致電子郵件無法傳送。 因此,FALSE
將協(xié)助您指明正確的方向,而 TRUE
並不一定表示您的電子郵件已成功傳送。值得注意的是!
許多共享網(wǎng)站主機,尤其是免費網(wǎng)站主機供應(yīng)商,要么不允許從其伺服器發(fā)送電子郵件,要么限制在任何給定時間段內(nèi)可以發(fā)送的數(shù)量。這是因為他們努力限制垃圾郵件發(fā)送者利用其更便宜的服務(wù)。
如果您認為您的主機有電子郵件限製或阻止發(fā)送電子郵件,請查看他們的常見問題解答,看看他們是否列出了任何此類限制。否則,您可能需要聯(lián)絡(luò)他們的支援人員,以驗證發(fā)送電子郵件是否有任何限制。
通常,由於各種原因,透過 PHP(和其他伺服器端程式語言)發(fā)送的電子郵件最終會進入收件者的垃圾郵件資料夾。在對代碼進行故障排除之前,請務(wù)必檢查那裡。
為了避免透過PHP 發(fā)送的郵件被傳送到收件者的垃圾郵件資料夾,您可以在PHP 程式碼或其他方面執(zhí)行多種操作,以最大程度地減少電子郵件被標(biāo)記為垃圾郵件的機會。 Michiel de Mare 提供的好建議包括:
有關(guān)此主題的詳細信息,請參閱如何確保以程式設(shè)計方式發(fā)送的電子郵件不會自動標(biāo)記為垃圾郵件? 。
如果郵件缺少「寄件者」和「回覆」等通用標(biāo)頭,某些垃圾郵件軟體將拒絕郵件:
$headers = array("From: from@example.com", "Reply-To: replyto@example.com", "X-Mailer: PHP/" . PHP_VERSION ); $headers = implode("\r\n", $headers); mail($to, $subject, $message, $headers);
無效的標(biāo)頭與沒有標(biāo)頭一樣糟糕。一個不正確的字元就可能使您的電子郵件脫軌。請仔細檢查以確保您的語法正確,因為 PHP 不會為您捕獲這些錯誤。
$headers = array("From from@example.com", // missing colon "Reply To: replyto@example.com", // missing hyphen "X-Mailer: "PHP"/" . PHP_VERSION // bad quotes );
寄件者:
寄件者雖然郵件必須有寄件者:寄件者,但您不能只使用任何值。特別是用戶提供的寄件者地址是阻止郵件的可靠方法:
$headers = array("From: $_POST[contactform_sender_email]"); // No!
原因:您的網(wǎng)路或發(fā)送郵件伺服器未列入 SPF/DKIM 白名單,無法假裝對 @hotmail 或 @gmail 位址負責(zé)。它甚至可能會默默地丟棄未配置的 From:
寄件人域的郵件。
有時,問題就像電子郵件收件者的值不正確一樣簡單。這可能是由於使用了不正確的變數(shù)。
$to = 'user@example.com'; // other variables .... mail($recipient, $subject, $message, $headers); // $recipient should be $to
測試此問題的另一種方法是將收件者值硬編碼到 mail()
函數(shù)呼叫中:
mail('user@example.com', $subject, $message, $headers);
這可以套用於所有 mail()
參數(shù)。
為了協(xié)助排除電子郵件帳戶問題,請將您的電子郵件傳送到不同電子郵件提供者的多個電子郵件帳戶。 。如果您的電子郵件未到達使用者的 Gmail 帳戶,請將相同的電子郵件傳送至 Yahoo 帳戶、Hotmail 帳戶和常規(guī) POP3 帳戶(例如您的 ISP 提供的電子郵件帳戶)。
如果電子郵件到達所有或部分其他電子郵件帳戶,則您知道您的程式碼正在發(fā)送電子郵件,但電子郵件帳戶提供者可能出於某種原因阻止了它們。如果電子郵件未到達任何電子郵件帳戶,則問題很可能與您的程式碼有關(guān)。
如果您已將表單方法設(shè)定為 POST
,請確保使用 $_POST
來尋找表單值。如果您已將其設(shè)定為 GET
或根本沒有設(shè)置,請確保使用 $_GET
來尋找表單值。
action
值指向正確的位置確保您的表單 action
屬性包含指向您的 PHP 郵件程式碼的值。