加油加油。
--WZY
一、發送郵件的原理
在了解其原理之前,先要知道兩個協議,SMTP和POP3
SMTP:Simple Mail Transfer Protocol,即簡單郵件傳輸協議,發送郵件的協議,默認端口,25
POP3:Post Office Protocol 3,即郵局協議,接收郵件的協議,默認端口,110
知道了這兩個協議,下面來說說郵件發送和接收的原理圖,分兩種,相同郵件(QQ郵箱給QQ郵箱)和不同郵件(QQ郵箱給163郵箱)是不一樣的。
解釋:
為了方便起見,將新浪郵箱,list@sina.com稱為用戶A,將搜狐郵箱,wangwu@sohu.com 稱為用戶B
用戶A(新浪郵箱,list@sina.com)給用戶B(搜狐郵箱,wangwu@sohu.com)發送一封郵件,那么整個過程就為圖中的實線部分,用戶A通過OutLook(瀏覽器或者客戶端)登錄自己的郵箱帳號,編寫郵件,使用Smtp協議發送給Sina的郵件服務器中的Smtp服務器(專門用來發送的服務器),然后在通過SMTP協議,傳輸給Sohu的郵件服務器中的Smtp服務器,然后通過Sohu的Smtp服務器將郵件發送給用戶B的存儲設備進行存儲(每個用戶都會有一個存儲空間用來存儲郵件的),到這里,用戶A就相當於發送成功了,因為已經到達了目的地,如果B用戶需要查看郵件內容,必須通過POP3服務器將從自己的存儲設備中拿到,然后返回到瀏覽器或者客戶端中顯示。
用戶B給用戶A發送一封郵件,那么整個過程就為圖中的虛線部分,是和A給B發一樣的步驟
用戶A給一個同樣使用新浪郵箱的用戶C發送一封郵件,那么其過程就簡單的多了,先通過Smtp服務器,然后smtp服務器會將其發送到用戶C的存儲設備上,A發送郵件就成功了,用戶C要顯示自己郵箱中的郵件,那么就通過POP3服務器從自己存儲設備中拿取所有郵件進行查看。
二、通過Java代碼實現發送郵件
2.1、准備jar包
核心:mail.jar
依賴:activation.jar, 郵件需要發送附件時使用
2.2、使用163郵箱發送郵件
2.2.1、首先在163郵箱中注冊一個帳號。有的話就省略這一步
2.2.2、編寫java代碼發送郵件,記住三大步
2.2.2.1、獲得連接(連接郵箱的服務器和登錄)
代碼

1 //0.1 確定連接位置 2 Properties props = new Properties(); 3 //獲取163郵箱smtp服務器的地址, 4 props.setProperty("mail.host", "smtp.163.com"); 5 //是否進行權限驗證。 6 props.setProperty("mail.smtp.auth", "true"); 7 8 9 //0.2確定權限(賬號和密碼) 10 Authenticator authenticator = new Authenticator() { 11 @Override 12 public PasswordAuthentication getPasswordAuthentication() { 13 //填寫自己的163郵箱的登錄帳號和授權密碼,授權密碼的獲取,在后面會進行講解。 14 return new PasswordAuthentication("163郵箱帳號","授權碼"); 15 } 16 }; 17 18 //1 獲得連接 19 /** 20 * props:包含配置信息的對象,Properties類型 21 * 配置郵箱服務器地址、配置是否進行權限驗證(帳號密碼驗證)等 22 * 23 * authenticator:確定權限(帳號和密碼) 24 * 25 * 所以就要在上面構建這兩個對象。 26 */ 27 28 Session session = Session.getDefaultInstance(props, authenticator);
2.2.2.2、創建消息(1、發件人,2、收件人,3、郵件標題,4、郵件內容)
代碼

1 //2 創建消息 2 Message message = new MimeMessage(session); 3 // 2.1 發件人 xxx@163.com 4 message.setFrom(new InternetAddress("xxx@163.com")); 5 /** 6 * 2.2 收件人 7 * 第一個參數: 8 * RecipientType.TO 代表收件人 9 * RecipientType.CC 抄送 10 * RecipientType.BCC 暗送 11 * 比如A要給B發郵件,但是A覺得有必要給要讓C也看看其內容,就在給B發郵件時, 12 * 將郵件內容抄送給C,那么C也能看到其內容了,但是B也能知道A給C抄送過該封郵件 13 * 而如果是暗送(密送)給C的話,那么B就不知道A給C發送過該封郵件。 14 * 第二個參數 15 * 收件人的地址,或者是一個Address[],用來裝抄送或者暗送人的名單。或者用來群發。可以是相同郵箱服務器的,也可以是不同的 16 * 這里我們發送給我們的qq郵箱 17 */ 18 message.setRecipient(RecipientType.TO, new InternetAddress("郵箱")); 19 // 2.3 主題(標題) 20 message.setSubject("郵件的標題"); 21 // 2.4 正文 22 String str = "李四: <br/>" + 23 "您好,您在本論壇注冊用戶,點擊下面url進行激活<br/>" + 24 "http://ww......<br/>" + 25 "如果不能點擊,請復制直接激活<br/>" + 26 "如果不是本人,請刪除郵件"; 27 //設置編碼,防止發送的內容中文亂碼。 28 message.setContent(str, "text/html;charset=UTF-8");
2.2.2.3、發送郵件
代碼

1 //3發送消息 2 Transport.send(message);
完整代碼

1 package javamail; 2 3 import java.util.Properties; 4 5 import javax.mail.Authenticator; 6 import javax.mail.Message; 7 import javax.mail.MessagingException; 8 import javax.mail.PasswordAuthentication; 9 import javax.mail.Session; 10 import javax.mail.Transport; 11 import javax.mail.Message.RecipientType; 12 import javax.mail.internet.AddressException; 13 import javax.mail.internet.InternetAddress; 14 import javax.mail.internet.MimeMessage; 15 16 public class Mail163Test { 17 public static void main(String[] args) throws Exception{ 18 //0.1 確定連接位置 19 Properties props = new Properties(); 20 //獲取163郵箱smtp服務器的地址, 21 props.setProperty("mail.host", "smtp.163.com"); 22 //是否進行權限驗證。 23 props.setProperty("mail.smtp.auth", "true"); 24 25 26 //0.2確定權限(賬號和密碼) 27 Authenticator authenticator = new Authenticator() { 28 @Override 29 public PasswordAuthentication getPasswordAuthentication() { 30 //填寫自己的163郵箱的登錄帳號和授權密碼,授權密碼的獲取,在后面會進行講解。 31 return new PasswordAuthentication("163郵箱的帳號","授權碼"); 32 } 33 }; 34 35 //1 獲得連接 36 /** 37 * props:包含配置信息的對象,Properties類型 38 * 配置郵箱服務器地址、配置是否進行權限驗證(帳號密碼驗證)等 39 * 40 * authenticator:確定權限(帳號和密碼) 41 * 42 * 所以就要在上面構建這兩個對象。 43 */ 44 45 Session session = Session.getDefaultInstance(props, authenticator); 46 47 48 //2 創建消息 49 Message message = new MimeMessage(session); 50 // 2.1 發件人 xxx@163.com 我們自己的郵箱地址,就是名稱 51 message.setFrom(new InternetAddress("xxx@163.com")); 52 /** 53 * 2.2 收件人 54 * 第一個參數: 55 * RecipientType.TO 代表收件人 56 * RecipientType.CC 抄送 57 * RecipientType.BCC 暗送 58 * 比如A要給B發郵件,但是A覺得有必要給要讓C也看看其內容,就在給B發郵件時, 59 * 將郵件內容抄送給C,那么C也能看到其內容了,但是B也能知道A給C抄送過該封郵件 60 * 而如果是暗送(密送)給C的話,那么B就不知道A給C發送過該封郵件。 61 * 第二個參數 62 * 收件人的地址,或者是一個Address[],用來裝抄送或者暗送人的名單。或者用來群發。可以是相同郵箱服務器的,也可以是不同的 63 * 這里我們發送給我們的qq郵箱 64 */ 65 message.setRecipient(RecipientType.TO, new InternetAddress("526745683@qq.com")); 66 // 2.3 主題(標題) 67 message.setSubject("郵件的標題"); 68 // 2.4 正文 69 String str = "李四: <br/>" + 70 "您好,您在本論壇注冊用戶,點擊下面url進行激活<br/>" + 71 "http://ww......<br/>" + 72 "如果不能點擊,請復制直接激活<br/>" + 73 "如果不是本人,請刪除郵件"; 74 //設置編碼,防止發送的內容中文亂碼。 75 message.setContent(str, "text/html;charset=UTF-8"); 76 77 78 //3發送消息 79 Transport.send(message); 80 } 81 }
2.2.3、授權碼的解釋
什么是授權碼?這個很簡單,如果你不是在163的郵箱網址中登錄的,而是在一些第三方客戶端中登錄郵箱帳號密碼,那么就必須進行授權,來獲取一個授權碼,使用該授權碼在第三方客戶端中進行登錄。而該授權碼就相當於我們的密碼了,帳號是不變的,也就是我們說我們在java代碼中想登錄郵箱,那么就必須使用授權碼進行登錄。如果獲取授權碼呢?
在網頁中登錄我們的163郵箱
登錄之后肯定顯示的是首頁,在首頁中有一個設置,點擊設置,在左面就會顯示一系列的設置選項,點擊客戶端授權密碼,就會顯示如圖中所示,讓其設置客戶端授權碼,如果沒開啟的話,默認就是關閉的,然后在按照步驟點擊開啟,一步步設置,就能得到授權碼。
如果不使用授權碼,那么java程序會報錯,報錯信息是驗證失敗,說明帳號密碼不正確,此時就是密碼不正確,因為這里密碼應該寫授權碼
2.2.4、成功發送郵件。
2.3、使用QQ郵箱發送郵件
2.3.1、首先在QQ郵箱中注冊一個帳號。有的話就省略這一步
2.3.2、進行三大步,創建連接、創建消息,發送消息
步驟都是一樣的,登錄時也是使用授權碼登錄(獲得授權碼跟163類似,先登錄,后找到設置,里面就有設置授權碼的步驟),但是你會發現,寫完代碼后會報錯,報錯信息如下
Exception in thread "main" javax.mail.AuthenticationFailedException: 530 Error: A secure connection is requiered(such as ssl). More information at http://service.mail.qq.com/cgi-bin/help?id=28
AuthenticationFailedException出現這個權限驗證失敗異常,看異常信息,說一個安全的連接要像ssl這樣被要求,更多的信息訪問這個網址:http://service.mail.qq.com/cgi-bin/help?id=28
訪問后,發現只有一個跟SSL相關的問題
點擊進去,會發現是關於第三方客戶端設置SSL加密的流程圖,但是我們可以知道我們出錯的原因是QQ郵箱發送或者接收郵件進行SSL加密。所以我就百度了一下,只需要加這四行代碼就搞定了
代碼

1 //QQ郵箱的SSL加密。 2 MailSSLSocketFactory sf = new MailSSLSocketFactory(); 3 sf.setTrustAllHosts(true); 4 props.put("mail.smtp.ssl.enable", "true"); 5 props.put("mail.smtp.ssl.socketFactory", sf);
mail.jar中其實已經封裝了SSL加密的算法,只需要拿出來用即可。
完整代碼

1 package javamail; 2 3 import java.security.GeneralSecurityException; 4 import java.util.Properties; 5 6 import javax.mail.Authenticator; 7 import javax.mail.Message; 8 import javax.mail.MessagingException; 9 import javax.mail.PasswordAuthentication; 10 import javax.mail.Session; 11 import javax.mail.Transport; 12 import javax.mail.Message.RecipientType; 13 import javax.mail.internet.AddressException; 14 import javax.mail.internet.InternetAddress; 15 import javax.mail.internet.MimeMessage; 16 17 import com.sun.mail.util.MailSSLSocketFactory; 18 19 public class QQMailTest { 20 21 public static void main(String[] args) throws AddressException, MessagingException, GeneralSecurityException { 22 23 //0.5,props和authenticator參數 24 Properties props = new Properties(); 25 props.setProperty("mail.host", "smtp.qq.com"); 26 props.setProperty("mail.smtp.auth", "true"); 27 28 //QQ郵箱的SSL加密。 29 MailSSLSocketFactory sf = new MailSSLSocketFactory(); 30 sf.setTrustAllHosts(true); 31 props.put("mail.smtp.ssl.enable", "true"); 32 props.put("mail.smtp.ssl.socketFactory", sf); 33 34 //authenticator參數,登錄自己的郵箱帳號密碼, 35 Authenticator authenticator = new Authenticator() { 36 @Override 37 public PasswordAuthentication getPasswordAuthentication() { 38 /** 39 * 注意,QQ郵箱的規則是如果不是由騰訊的網頁或者客戶端打開登錄的話,在其他任何地方 40 *登錄郵箱,密碼必須使用授權碼,授權碼下面會講解,vlyvawibbsribgee 41 *xxxxxxx:自己的QQ郵箱登錄帳號,也就是qq號 42 *yyyyyyy:密碼,使用授權碼登錄,而不能使用原始的QQ密碼 43 */ 44 return new PasswordAuthentication("qq郵箱帳號","授權碼"); 45 } 46 }; 47 //1、連接 48 /** 49 * props 50 * 連接配置信息,郵件服務器的地址,是否進行權限驗證 51 * authenticator 52 * 權限驗證,也就是帳號密碼驗證 53 * 所以需要先配置這兩個參數 54 */ 55 Session session = Session.getDefaultInstance(props, authenticator); 56 57 //2、發送的內容對象Mesage 58 Message message = new MimeMessage(session); 59 //2.1、發件人是誰 60 message.setFrom(new InternetAddress("526745683@qq.com")); 61 // 2.2 , to:收件人 ; cc:抄送 ; bcc :暗送. 62 /** 63 * 收件人是誰? 64 * 第一個參數: 65 * RecipientType.TO 代表收件人 66 * RecipientType.CC 抄送 67 * RecipientType.BCC 暗送 68 * 比如A要給B發郵件,但是A覺得有必要給要讓C也看看其內容,就在給B發郵件時, 69 * 將郵件內容抄送給C,那么C也能看到其內容了,但是B也能知道A給C抄送過該封郵件 70 * 而如果是暗送(密送)給C的話,那么B就不知道A給C發送過該封郵件。 71 * 第二個參數 72 * 收件人的地址,或者是一個Address[],用來裝抄送或者暗送人的名單。或者用來群發。 73 */ 74 message.setRecipient(RecipientType.TO, new InternetAddress("收件人的郵箱地址 xxx@qq.com")); 75 // 2.3 主題(標題) 76 message.setSubject("hello"); 77 // 2.4 正文 78 String str = "苦水潤喉: <br/>" + 79 "hah<br/>"; 80 message.setContent(str, "text/html;charset=UTF-8"); 81 //3、發送 82 Transport.send(message); 83 } 84 85 }
2.4、總結
上面使用QQ郵箱和163郵箱發送郵件, 其他的一些郵箱服務器,應該也類似,就算有也是一些細微的區別,就好比上面QQ郵箱需要使用SSL加密,而163不需要一樣,遇到錯誤可以百度自己解決一下。然后必須注明一點,上面的發送郵件的前提是,必須要聯網,如果不能聯網,那就不能夠成功發送郵件,並且會報異常,一看這圖,就應該明白為什么需要連網,因為不連網,我們連服務器地址都解析不出,就更不能讓郵件傳送了。
就怕有些人不能連網,那么就不能判斷自己編寫的代碼是否正確,也不能體驗到這種能夠自己發送郵件的的感覺了。所以下面就介紹在本機上安裝郵箱服務器和客戶端,那么就不用連網就能夠收發郵件了。
三、安裝郵箱服務器和客戶端
3.1、郵箱服務器和客戶端的類型
郵箱服務器:eyoumailserverstup.exe 下載地址可以百度一下,CSDN論壇上也有該資源
客戶端:Foxmail_7.0.exe 跟上面同理
3.2、安裝郵箱服務器
安裝完后會出現一個
這個不用管它,不影響使用,安裝后的樣子
1、將Admin刪除,並且點擊設置,設置我們自定義的域名
2、點擊新帳號,創建兩個帳號,一個wuhao、一個zhangsan
3、上面的操作什么意思呢?就相當於我自己創建了一個郵箱服務器,並且在該服務器上注冊了兩個用戶,好比兩個人在163郵箱中注冊兩個帳號。
4、測試,wuhao給zhangsan發郵件
代碼根據163的那套,不用使用ssl加密,也不用使用授權碼,直接使用我們的登錄密碼
郵箱服務器地址:127.0.0.1 就是本機地址
帳號密碼:wuhao 123
發件人地址:wuhao@wh.cn wh.cn使我們在服務器中設置的域名
收件人地址:zhangsan@wh.cn
關鍵參數在上面已經說完了,結果如何呢?在服務器中zhangsan這里確實收到了一封郵件,並且看下面注釋也能知道,確實是wuhao發送給zhangsan的郵件。
5、因為這是服務器,是不可以看到郵件內容的,就像我們一開始說的原理圖,要想看到郵件內容,則需要安裝客戶端,通過pop3協議來查看。
3.3、安裝第三方郵箱客戶端
Foxmail7.0
3.3.1、安裝后,設置郵箱地址,我們需要查看獲取zhangsan的郵件,所以填寫zhangsan@wh.cn,因為這個服務器是我們自己開的,所以並沒有提示。
3.3.2、設置帳號密碼,和郵箱類型,郵箱類型選擇POP3,用來接收郵件協議的
3.3.3、其中接收和發送郵件服務器,都應該寫本機地址127.0.0.1,因為我們本機就是一個郵箱服務器,而如果是別的郵箱服務器的話,那就不一樣了,比如sohu郵箱服務器,比如qq郵箱服務器,163郵箱服務器類似這種就應該使用pop.163.com,smtp.163.com來獲取服務器地址。並且這里注意到沒有,有一個使用ssl來接連服務器的選項,記得我們使用qq郵箱時遇到的問題嗎,對,如果你綁定的是你的QQ郵箱,那么這里就需要將次選項進行勾選。
3.3.4、設置完成后,就顯示出了zhangsan有一張新郵件
四、總結
用javamail發送郵件,到這里就全部講解完了,你學會了嗎?不過這種技術現在都不怎么用了,用的比較多的還是直接用手機號碼進行驗證,而不是用郵箱。不過學會了還是不錯的,比如連續給一個人發送100封郵件。哈哈,不太好吧,總之希望這篇文章對大家有所幫助。也希望大家都能夠自己手動實現該功能,自己玩一玩。