加油加油。html
--WZYjava
1、發送郵件的原理算法
在瞭解其原理以前,先要知道兩個協議,SMTP和POP3瀏覽器
SMTP:Simple Mail Transfer Protocol,即簡單郵件傳輸協議,發送郵件的協議,默認端口,25安全
POP3:Post Office Protocol 3,即郵局協議,接收郵件的協議,默認端口,110服務器
知道了這兩個協議,下面來講說郵件發送和接收的原理圖,分兩種,相同郵件(QQ郵箱給QQ郵箱)和不一樣郵件(QQ郵箱給163郵箱)是不同的。session
解釋: socket
爲了方便起見,將新浪郵箱,list@sina.com稱爲用戶A,將搜狐郵箱,wangwu@sohu.com 稱爲用戶Bide
用戶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服務器從本身存儲設備中拿取全部郵件進行查看。
2、經過Java代碼實現發送郵件
2.一、準備jar包
核心:mail.jar
依賴:activation.jar, 郵件須要發送附件時使用
2.二、使用163郵箱發送郵件
2.2.一、首先在163郵箱中註冊一個賬號。有的話就省略這一步
2.2.二、編寫java代碼發送郵件,記住三大步
2.2.2.一、得到鏈接(鏈接郵箱的服務器和登陸)
代碼
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.二、建立消息(一、發件人,二、收件人,三、郵件標題,四、郵件內容)
代碼
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.三、發送郵件
代碼
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.三、受權碼的解釋
什麼是受權碼?這個很簡單,若是你不是在163的郵箱網址中登陸的,而是在一些第三方客戶端中登陸郵箱賬號密碼,那麼就必須進行受權,來獲取一個受權碼,使用該受權碼在第三方客戶端中進行登陸。而該受權碼就至關於咱們的密碼了,賬號是不變的,也就是咱們說咱們在java代碼中想登陸郵箱,那麼就必須使用受權碼進行登陸。若是獲取受權碼呢?
在網頁中登陸咱們的163郵箱
登陸以後確定顯示的是首頁,在首頁中有一個設置,點擊設置,在左面就會顯示一系列的設置選項,點擊客戶端受權密碼,就會顯示如圖中所示,讓其設置客戶端受權碼,若是沒開啓的話,默認就是關閉的,而後在按照步驟點擊開啓,一步步設置,就能獲得受權碼。
若是不使用受權碼,那麼java程序會報錯,報錯信息是驗證失敗,說明賬號密碼不正確,此時就是密碼不正確,由於這裏密碼應該寫受權碼
2.2.四、成功發送郵件。
2.三、使用QQ郵箱發送郵件
2.3.一、首先在QQ郵箱中註冊一個賬號。有的話就省略這一步
2.3.二、進行三大步,建立鏈接、建立消息,發送消息
步驟都是同樣的,登陸時也是使用受權碼登陸(得到受權碼跟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 //一、鏈接 48 /** 49 * props 50 * 鏈接配置信息,郵件服務器的地址,是否進行權限驗證 51 * authenticator 52 * 權限驗證,也就是賬號密碼驗證 53 * 因此須要先配置這兩個參數 54 */ 55 Session session = Session.getDefaultInstance(props, authenticator); 56 57 //二、發送的內容對象Mesage 58 Message message = new MimeMessage(session); 59 //2.一、發件人是誰 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 //三、發送 82 Transport.send(message); 83 } 84 85 }
2.四、總結
上面使用QQ郵箱和163郵箱發送郵件, 其餘的一些郵箱服務器,應該也相似,就算有也是一些細微的區別,就比如上面QQ郵箱須要使用SSL加密,而163不須要同樣,遇到錯誤能夠百度本身解決一下。而後必須註明一點,上面的發送郵件的前提是,必需要聯網,若是不能聯網,那就不可以成功發送郵件,而且會報異常,一看這圖,就應該明白爲何須要連網,由於不連網,咱們連服務器地址都解析不出,就更不能讓郵件傳送了。
就怕有些人不能連網,那麼就不能判斷本身編寫的代碼是否正確,也不能體驗到這種可以本身發送郵件的的感受了。因此下面就介紹在本機上安裝郵箱服務器和客戶端,那麼就不用連網就可以收發郵件了。
3、安裝郵箱服務器和客戶端
3.一、郵箱服務器和客戶端的類型
郵箱服務器:eyoumailserverstup.exe 下載地址能夠百度一下,CSDN論壇上也有該資源
客戶端:Foxmail_7.0.exe 跟上面同理
3.二、安裝郵箱服務器
安裝完後會出現一個
這個不用管它,不影響使用,安裝後的樣子
一、將Admin刪除,而且點擊設置,設置咱們自定義的域名
二、點擊新賬號,建立兩個賬號,一個wuhao、一個zhangsan
三、上面的操做什麼意思呢?就至關於我本身建立了一個郵箱服務器,而且在該服務器上註冊了兩個用戶,比如兩我的在163郵箱中註冊兩個賬號。
四、測試,wuhao給zhangsan發郵件
代碼根據163的那套,不用使用ssl加密,也不用使用受權碼,直接使用咱們的登陸密碼
郵箱服務器地址:127.0.0.1 就是本機地址
賬號密碼:wuhao 123
發件人地址:wuhao@wh.cn wh.cn使咱們在服務器中設置的域名
收件人地址:zhangsan@wh.cn
關鍵參數在上面已經說完了,結果如何呢?在服務器中zhangsan這裏確實收到了一封郵件,而且看下面註釋也能知道,確實是wuhao發送給zhangsan的郵件。
五、由於這是服務器,是不能夠看到郵件內容的,就像咱們一開始說的原理圖,要想看到郵件內容,則須要安裝客戶端,經過pop3協議來查看。
3.三、安裝第三方郵箱客戶端
Foxmail7.0
3.3.一、安裝後,設置郵箱地址,咱們須要查看獲取zhangsan的郵件,因此填寫zhangsan@wh.cn,由於這個服務器是咱們本身開的,因此並無提示。
3.3.二、設置賬號密碼,和郵箱類型,郵箱類型選擇POP3,用來接收郵件協議的
3.3.三、其中接收和發送郵件服務器,都應該寫本機地址127.0.0.1,由於咱們本機就是一個郵箱服務器,而若是是別的郵箱服務器的話,那就不同了,好比sohu郵箱服務器,好比qq郵箱服務器,163郵箱服務器相似這種就應該使用pop.163.com,smtp.163.com來獲取服務器地址。而且這裏注意到沒有,有一個使用ssl來接連服務器的選項,記得咱們使用qq郵箱時遇到的問題嗎,對,若是你綁定的是你的QQ郵箱,那麼這裏就須要將次選項進行勾選。
3.3.四、設置完成後,就顯示出了zhangsan有一張新郵件
4、總結
用javamail發送郵件,到這裏就所有講解完了,你學會了嗎?不過這種技術如今都不怎麼用了,用的比較多的仍是直接用手機號碼進行驗證,而不是用郵箱。不過學會了仍是不錯的,好比連續給一我的發送100封郵件。哈哈,不太好吧,總之但願這篇文章對你們有所幫助。也但願你們都可以本身手動實現該功能,本身玩一玩。