dyerac  
          dyerac In Java
          公告

          日歷
          <2025年7月>
          293012345
          6789101112
          13141516171819
          20212223242526
          272829303112
          3456789
          統(tǒng)計(jì)
          • 隨筆 - 36
          • 文章 - 10
          • 評(píng)論 - 94
          • 引用 - 0

          導(dǎo)航

          常用鏈接

          留言簿(5)

          隨筆分類(49)

          隨筆檔案(36)

          文章分類(11)

          文章檔案(10)

          相冊(cè)

          dyerac

          搜索

          •  

          積分與排名

          • 積分 - 79674
          • 排名 - 705

          最新隨筆

          最新評(píng)論

          閱讀排行榜

          評(píng)論排行榜

           
          ?? 作為企業(yè)級(jí)應(yīng)用的JavaEE,郵件收發(fā)毫無疑問是其重要技術(shù)組成;在這方面,JavaMail庫和Apache的通用電子郵件軟件包給我們提供了兩個(gè)選擇.不過通用電子郵件庫實(shí)際上是包裹在JavaMail外層的API,所以無論我們選擇哪種API,都需要JavaMail庫;我們可能還需要JavaBeans激活框架(JavaBeans Activation Framework(JAF)),該框架將負(fù)責(zé)處理關(guān)于郵件選項(xiàng)的更復(fù)雜的內(nèi)容.由于通用電子郵件軟件包并沒有實(shí)現(xiàn)收取郵件的操作,在這里,我們暫且只討論JavaMail的實(shí)現(xiàn).

          一.郵件的發(fā)送
          ? 第一件要知道的事情是,你的SMTP服務(wù)器的主機(jī)名,它負(fù)責(zé)將您的郵件發(fā)送到外部世界的機(jī)器.一般來說這些服務(wù)器都符合命名習(xí)慣,比如,如果你的郵箱是acmilan@sina.com.cn,那么SMTP服務(wù)器的主機(jī)名則是smtp.sina.com.cn;另外也可以參考各大網(wǎng)站自己的說明.為了方便,下文中以網(wǎng)易郵箱為例.
          ? JavaMail使用了Session類的概念來保存諸如SMTP主機(jī)和認(rèn)證的信息,主要想法是基于會(huì)話(Sessions)在Java虛擬機(jī)中可以被隔離,這可以阻止惡意代碼竊取其他用戶在其他會(huì)話中的信息,這些信息可能包括用戶名和密碼等認(rèn)證信息.你所要發(fā)送的郵件將保存在一個(gè)Message對(duì)象中,而這個(gè)Message對(duì)象則是由你所構(gòu)造的session實(shí)例來創(chuàng)建
          ? 要得到一個(gè)特定的session對(duì)象,可以通過一下代碼:
          ????? //設(shè)置session的屬性
          ????? Properties pro = new Properties();
          ????? pro.put("mail.transport.protocol", "smtp");
          ????? pro.put("mail.smtp.auth", "true");
          ????? pro.put("mail.smtp.host", "smtp.126.com");
          ????? pro.put("mail.host", "126.com");
          ?????
          ????? //設(shè)置認(rèn)證器
          ????? PopupAuthenticator pop = new PopupAuthenticator();
          ????? pop.performCheck("My Name", "My Password");//你的帳戶和密碼
          ?????
          ????? //得到session
          ????? Session mailSession = Session.getInstance(pro, pop);
          ? 要注意的是,為了避免垃圾郵件,大多數(shù)的smtp服務(wù)器需要認(rèn)證,SMTP認(rèn)證(SMTP AUTH)需要用戶名和密碼來發(fā)送郵件;因此,必須在session的初始化參數(shù)中設(shè)置一個(gè)認(rèn)證者(Authenticator)來返回所需的認(rèn)證證書,具體代碼必須由自己來實(shí)現(xiàn):
          ???? class PopupAuthenticator extends Authenticator {
          ?????? String username = null;
          ?????? String password = null;
          ?????? public PopupAuthenticator() {}
          ?????? public PasswordAuthentication performCheck(String user, String pass) {
          ???????? username = user;
          ???????? password = pass;
          ???????? return getPasswordAuthentication();
          ??????? }
          ?????? protected PasswordAuthentication getPasswordAuthentication() {
          ???????? return new PasswordAuthentication(username, password);
          ??????? }
          ????? }

          ? 接著,就可以用之前得到的session來構(gòu)造Message對(duì)象:
          ??? Message msg = new MimeMessage(mailSession);?
          ? 在使用會(huì)話創(chuàng)建了一個(gè)MimeMessage后,我們需要來填充這個(gè)消息.首先是設(shè)置表頭信息,Message類定義了郵件系統(tǒng)中使用的屬性,由名字-值對(duì)組成,使用這些名字-值可以指定郵件表頭信息,Javamail提供了一系列api用于設(shè)置常見的郵件表頭,其中在涉及地址的操作時(shí),我們用InternetAddress來進(jìn)行封裝:
          ????? msg.setFrom(new InternetAddress ("acmilan@126.com");
          ????? msg.setSubject("Hello");
          ????? msg.setRecipients(Message.RecipientType.TO, InternetAddress.parse("intermilan@126.com", false));
          ????? msg.setText("I will beat u");
          ????? msg.setSentDate(new Date());
          ????? //發(fā)送消息
          ????? Transport.send(msg);
          ? 對(duì)Transport類的調(diào)用將會(huì)去查找適當(dāng)?shù)臅?huì)話,并找出如何發(fā)送消息,盡管這樣做看上去有些不直觀。當(dāng)我們完成這一步的時(shí)候,我們的郵件就已經(jīng)發(fā)送出去了。此時(shí),我們還需要添加代碼來捕獲三種JavaMail可能拋出的異常,它們是AddressException、MessagingException和UnsupportedEncodingException. 但這就是最基本的使用JavaMail發(fā)送郵件的方法。
          ? 有時(shí)候我們還需要給郵件添加附件.再回到之前對(duì)Message的討論中,Message對(duì)象同樣定義了郵件的內(nèi)容,它可以定義一個(gè)消息內(nèi)容,也可以定義多個(gè)消息內(nèi)容,消息內(nèi)容(通常指的是附件)都將由DataHandle下的類來處理.Message對(duì)象由Multipart組成,一個(gè)Multipart可含有多個(gè)BodyPart,這些BodyPart將用來保存文本信息和附件.
          ????? MimeMultipart multipart = new MimeMultipart();
          ????? BodyPart msgBodyPart = new MimeBodyPart();//用來放置文本內(nèi)容
          ????? msgBodyPart.setContent(message, "text/plain");
          ????? BodyPart attBodyPart = new MimeBodyPart();//用來放置附件
          ????? DataSource ds = new FileDataSource(new File("c:/td.txt"));
          ????? attBodyPart.setDataHandler(new DataHandler(ds));//設(shè)置DataHandler
          ????? attBodyPart.setFileName("bsbs.txt");//附件的顯示名字
          ????? multipart.addBodyPart(msgBodyPart);
          ????? multipart.addBodyPart(attBodyPart);
          ????? msg.setContent(multipart);
          ????? Transport.send(msg);
          ?? 最后,我們來看看如何發(fā)送HTML格式的郵件,文本的格式必須相應(yīng)的設(shè)置為text/html,郵件中的圖片將以附件形式加載,另外還要指定一個(gè)內(nèi)部ID以供調(diào)用;
          ????? MimeMultipart multipart = new MimeMultipart();
          ????? BodyPart msgBodyPart = new MimeBodyPart();
          ????? //設(shè)置格式為"text/html"
          ????? msgBodyPart.setContent("<H1>Hi! From HtmlJavaMail</H1> <img src=\"cid:logo\"/>", "text/html");
          ????? BodyPart embedImage = new MimeBodyPart();
          ????? DataSource ds = new URLDataSource(new URL("????? embedImage.setDataHandler(new DataHandler(ds));
          ????? //設(shè)置表頭的內(nèi)部ID,注意,所設(shè)置內(nèi)容必須與前文對(duì)應(yīng),在此處,前文的引用為<img src=\"cid:logo\"/>,因此Content-ID表頭對(duì)應(yīng)
          ????? //的應(yīng)該是<logo>
          ????? embedImage.setHeader("Content-ID", "<logo>");
          ????? multipart.addBodyPart(msgBodyPart);
          ????? multipart.addBodyPart(embedImage);
          ????? msg.setContent(multipart);
          ?? 這樣,一封HTML格式的郵件便完成了
          ????? Transport.send(msg);


          二.郵件的收取
          ? 同樣,第一步還是要獲得服務(wù)器的名字.我們還是以網(wǎng)易為例.
          ? 接收郵件包含兩個(gè)協(xié)議,即POP3和IMAP。POP3是老協(xié)議,它提供一個(gè)單一收信箱,以存放一定順序的郵件信息。IMAP相對(duì)比較新,它為郵件提供連接到一個(gè)層次關(guān)系的文件入口,其中一個(gè)入口即為收信箱。當(dāng)然還有其它可以使用的協(xié)議,POP3和IMAP只是其中一種安全性很好的協(xié)議。JavaMail將這些協(xié)議提煉為一種郵件倉庫(Store)的概念,這一倉庫為文件等級(jí)的集合。這種提煉意味著倉庫包含很多內(nèi)容,但我們只需要弄清楚在一個(gè)服務(wù)文件夾中瀏覽與導(dǎo)航郵件信息的過程,而實(shí)際處理郵件協(xié)議的任務(wù)則通過JavaMail調(diào)用Provider來完成,不同的協(xié)議對(duì)應(yīng)不同的provider。
          ? JavaMail的Provider操作服務(wù)于POP3和IMAP,你也可以將另外的Providers嵌入到JavaMail API以處理其它諸如NNTP或本地存儲(chǔ)郵件的協(xié)議。在Sun主頁上列出這方面的第三方Providers。
          ? 在Javamail中,store和folder類是用來存儲(chǔ)和接受消息.store由具體的session得到,它使用可帶參數(shù)的connect方法與服務(wù)器連接,folder則和File有點(diǎn)類似,可以將其比作windows下的文件夾.客戶由store類中取得folder,再對(duì)folder進(jìn)行操作--進(jìn)入特定的folder,讀取folder中的消息.
          ???? Session session;
          ???? Store store = null;
          ???? Folder folder = null;
          ???? Folder inboxfolder = null;
          ????
          ???? Properties props = System.getProperties();
          ???? props.setProperty("mail.pop3s.rsetbeforequit", "true");//這樣讀取郵件時(shí)服務(wù)器不會(huì)刪除原有的郵件
          ???? props.setProperty("mail.pop3.rsetbeforequit", "true");
          ???? session = Session.getInstance(props, null);

          ???? store = session.getStore("pop3");//通過"pop3"得到適當(dāng)?shù)膒rovider
          ???? store.connect(emailserver, emailuser, emailpassword);
          ???? folder = store.getDefaultFolder();//得到默認(rèn)的頂級(jí)文件夾
          ? 每一Folder可以包含很多其它的文件夾,通過getFolder方法來瀏覽特定文件夾。INBOX是郵件服務(wù)器表示的主文件夾保留的名稱,pop3和imap存儲(chǔ)都會(huì)提供一個(gè)INBOX文件夾.
          ???? inboxfolder = folder.getFolder("INBOX");
          ???? inboxfolder.open(Folder.READ_ONLY);//只讀模式
          ???? Message[] msgs = inboxfolder.getMessages();
          ? 需要注意的是,此時(shí)真正的消息對(duì)象并沒有存儲(chǔ)到數(shù)組(msgs)中,數(shù)組保存的只是這些消息對(duì)象的引用,只有在調(diào)用了具體的Message.getXX()方法時(shí),程序才會(huì)再次連接服務(wù)器并取得真正的結(jié)果.如果要進(jìn)行對(duì)郵件的篩選工作,不停的調(diào)用每個(gè)消息對(duì)象的getXX方法無疑會(huì)影響性能.因此我們可以用FetchProfile類來預(yù)先抓取感興趣的部分內(nèi)容(如各表頭內(nèi)容),這就不需要每次調(diào)用getXX方法時(shí)都再連接服務(wù)器了.
          ???? FetchProfile fp = new FetchProfile();
          ???? fp.add("Subject");//即只讀取主題信息
          ???? inboxfolder.fetch(msgs, fp);//預(yù)讀取每個(gè)消息的主題
          ???? for (int j = 0 ;j <msgs.length; j++) {
          ??????? if (msgs[j].getSubject().startsWith("^_^")) {//只對(duì)標(biāo)題為"^_^"的郵件進(jìn)行操作
          ????????? .......
          ??????? }
          ? 接下來,就可以調(diào)用Message的各種方法對(duì)郵件進(jìn)行操作了.對(duì)不同格式的郵件,具體的操作當(dāng)然也略微不同.一個(gè)做法是對(duì)每個(gè)具體的BodyPart進(jìn)行操作,通過下列的遞歸方法可以獲得每個(gè)BodyPart的引用
          ???? private void extractPart(final Part part) throws MessagingException,IOException {
          ?????? if(part.getContent() instanceof Multipart) {
          ?????????? Multipart mp=(Multipart)part.getContent();
          ?????????? for (int i = 0; i < mp.getCount(); i++) {
          ?????????????? extractPart(mp.getBodyPart(i));
          ?????????? }
          ????? return;
          ??? }
          ? Part的getContentType方法可以為我們對(duì)其采取何種方法處理提供依據(jù)
          ???? part.getContentType().startsWith("text/plain").....
          ? 如果是以"text/plain"或者"text/html"開頭,通常我們可以直接取出其內(nèi)容
          ??? bodytext = (String) part.getContent();
          ? 如果兩者都不是,我們將其視之為附件,通過IO流來讀取:
          ????? InputStream in = part.getInputStream();//讀取part中的附件
          ????? ByteArrayOutputStream bos = new ByteArrayOutputStream();

          ????? byte[] buffer = new byte[8192];
          ????? int count = 0;
          ????? while ( (count = in.read(buffer)) >= 0) {
          ??????? bos.write(buffer, 0, count);
          ????? }
          ????? in.close();
          ? 接下來,就可以對(duì)流進(jìn)行操作了.
          ? 那么,關(guān)于Javamail的研究也就進(jìn)行到這兒了.

          參考文獻(xiàn):
          ? DJ Walker-Morgan????????? Getting the mail in: receiving in JavaMail
          ?????????????????????????????????????Sending email in Java: There's more than one way
          ? 趙強(qiáng),喬新亮???????????????????J2EE應(yīng)用開發(fā)


          只有注冊(cè)用戶登錄后才能發(fā)表評(píng)論。


          網(wǎng)站導(dǎo)航:
          博客園   IT新聞   Chat2DB   C++博客   博問  
           
           
          Copyright © dyerac in java... Powered by: 博客園 模板提供:滬江博客
          主站蜘蛛池模板: 普兰店市| 清河县| 宜兰县| 宁国市| 东丽区| 高唐县| 宁晋县| 太仓市| 六枝特区| 琼结县| 昌吉市| 满城县| 延川县| 子洲县| 雅江县| 波密县| 安康市| 淮滨县| 南通市| 玛纳斯县| 垫江县| 安福县| 正宁县| 建始县| 门源| 临邑县| 台东县| 云霄县| 岳池县| 元朗区| 新昌县| 金川县| 金塔县| 沁水县| 东台市| 济宁市| 建始县| 汶上县| 图们市| 灵丘县| 深水埗区|