JSP與Java Mail API
發(fā)表時(shí)間:2024-02-06 來(lái)源:明輝站整理相關(guān)軟件相關(guān)文章人氣:
[摘要]綜述:Java Mail API的開(kāi)發(fā)是SUN為Java開(kāi)發(fā)者提供公用API框架的持續(xù)努力的良好例證。提倡公用框架,反對(duì)受限于供應(yīng)商的解決方案,充分預(yù)示著一個(gè)日益開(kāi)放的開(kāi)發(fā)環(huán)境的建立! ava Mail API的結(jié)構(gòu)本身證明了它的開(kāi)發(fā)者的基本目標(biāo)之一--軟件開(kāi)發(fā)的工作量應(yīng)該取決于應(yīng)用程序本身的...
綜述:Java Mail API的開(kāi)發(fā)是SUN為Java開(kāi)發(fā)者提供公用API框架的持續(xù)努力的良好例證。提倡公用框架,反對(duì)受限于供應(yīng)商的解決方案,充分預(yù)示著一個(gè)日益開(kāi)放的開(kāi)發(fā)環(huán)境的建立。
Java Mail API的結(jié)構(gòu)本身證明了它的開(kāi)發(fā)者的基本目標(biāo)之一--軟件開(kāi)發(fā)的工作量應(yīng)該取決于應(yīng)用程序本身的復(fù)雜程度以及開(kāi)發(fā)者所要求的控制程度。換句話說(shuō),Java Mail API盡可能地保持簡(jiǎn)單。乍看起來(lái),Java Mail API所擁有的類總數(shù)以及類之間的關(guān)系可能讓人誤解為要花費(fèi)漫長(zhǎng)的學(xué)習(xí)時(shí)間。實(shí)際上,一旦正式開(kāi)始使用,你就會(huì)發(fā)現(xiàn)該API不失為在應(yīng)用程序中加入健壯的郵件/通訊支持的簡(jiǎn)單工具。
建立JavaMail使用環(huán)境,需要哪些軟件
首先,需要安裝JavaMail API,F(xiàn)在有兩種常用的 JavaMail API 版本:1.2 和 1.1.3。雖然版本 1.2 是最新版,但版本 1.1.3 包含了Java 2 平臺(tái)企業(yè)版(Java 2 Platform, Enterprise Edition,J2EE)的版本 1.2.1,所以仍然有很多人使用它。
·JavaMail 1.2 的安裝
要使用 JavaMail 1.2 API,請(qǐng)下載 JavaMail 1.2 實(shí)現(xiàn),解開(kāi)Javamail-1_2.zip 文件,并將 mail.jar 文件添加到 CLASSPATH 中。除了核心類,隨版本 1.2 實(shí)現(xiàn)一起提供的還有 SMTP、IMAP4 和 POP3 供應(yīng)商。
· JavaMail 1.1.3 的安裝
要使用 JavaMail 1.1.3 API,請(qǐng)下載 JavaMail 1.1.3 實(shí)現(xiàn),解開(kāi)Javamail1_1_3.zip文件,并將 mail.jar 文件添加到您的 CLASSPATH 中。除了核心類,隨版本 1.1.3 實(shí)現(xiàn)一起提供的還有 SMTP 和 IMAP4 供應(yīng)商。
如果您用 JavaMail 1.1.3 訪問(wèn)一個(gè) POP 服務(wù)器,請(qǐng)下載并安裝一個(gè) POP3 供應(yīng)商。Sun 就有一個(gè)獨(dú)立于 JavaMail 實(shí)現(xiàn)。下載并解開(kāi) pop31_1_1.zip 文件后,將 pop3.jar 也添加到您的 CLASSPATH 中。
其次是JavaBeans Activation Framework 的安裝。JavaMail API 的所有版本都需要 JavaBeans Activation Framework 來(lái)支持任意數(shù)據(jù)塊的輸入及相應(yīng)處理。功能似乎不多,但目前許多瀏覽器和郵件工具中都能找到這種基本的 MIME 型支持。下載完框架后,解開(kāi) jaf1_0_1.zip 文件,并將 activation.jar 文件添加到 CLASSPATH 中。
對(duì)于 JavaMail 1.2 用戶來(lái)說(shuō),現(xiàn)在您應(yīng)該已將 mail.jar 和 activation.jar 文件添加到 CLASSPATH 中了。
而對(duì)于 JavaMail 1.1.3 用戶,現(xiàn)在您應(yīng)該已將 mail.jar、pop3.jar和 activation.jar文件添加到 CLASSPATH 中了。如果不打算用 POP3,就不必將 pop3.jar 添加到 CLASSPATH 中去。
如果您不想更改 CLASSPATH 環(huán)境變量,將 jar 文件復(fù)制到您 Java 運(yùn)行時(shí)環(huán)境(Java Runtime Environment,JRE)目錄下的 lib/ext 目錄中去。例如,J2SE 1.3 發(fā)行版的缺省目錄在 Windows 平臺(tái)的 C:\jdk1.3\jre\lib\ext。
Java Mail API有哪些核心類
·Javax.mail.Session:Session 類定義了一個(gè)基本郵件會(huì)話(session),是Java Mail API最高層入口類。所有其它類都是經(jīng)由這個(gè)session 才得以生效。Session 對(duì)象用 Java.util.Properties 對(duì)象獲取信息,如郵件服務(wù)器、用戶名、密碼及整個(gè)應(yīng)用程序中共享的其它信息。
·Javax.mail.Message:一旦獲得 Session 對(duì)象,就可以繼續(xù)創(chuàng)建要發(fā)送的消息。這由 Message 類來(lái)完成。因?yàn)?Message 是個(gè)抽象類,必需用一個(gè)子類,多數(shù)情況下為 Javax.mail.internet.MimeMessage。MimeMessage 是個(gè)能理解 MIME 類型和頭的電子郵件消息,正如不同 RFC 中所定義的。雖然在某些頭部域非 ASCII 字符也能被譯碼,但 Message 頭只能被限制為用 US-ASCII 字符。
·Javax.mail.Address:一旦您創(chuàng)建了 Session 和 Message,并將內(nèi)容填入消息后,就可以用 Address 確定信件地址了。和 Message 一樣,Address 也是個(gè)抽象類。您用的是 Javax.mail.internet.InternetAddress 類。
·Javax.mail. Authenticator:與 Java.net 類一樣,JavaMail API 也可以利用 Authenticator 通過(guò)用戶名和密碼訪問(wèn)受保護(hù)的資源。對(duì)于JavaMail API 來(lái)說(shuō),這些資源就是郵件服務(wù)器。JavaMail Authenticator 在 Javax.mail 包中,而且它和 Java.net 中同名的類 Authenticator 不同。兩者并不共享同一個(gè) Authenticator,因?yàn)镴avaMail API 用于 Java 1.1,它沒(méi)有 Java.net 類別。
要使用 Authenticator,先創(chuàng)建一個(gè)抽象類的子類,并從 getPasswordAuthentication() 方法中返回 PasswordAuthentication 實(shí)例。創(chuàng)建完成后,您必需向 session 注冊(cè) Authenticator。然后,在需要認(rèn)證的時(shí)候,就會(huì)通知 Authenticator。您可以彈出窗口,也可以從配置文件中(雖然沒(méi)有加密是不安全的)讀取用戶名和密碼,將它們作為 PasswordAuthentication 對(duì)象返回給調(diào)用程序。
·Javax.mail.Transport:消息發(fā)送的最后一部分是使用 Transport 類。這個(gè)類用協(xié)議指定的語(yǔ)言發(fā)送消息(通常是 SMTP)。它是抽象類,它的工作方式與 Session 有些類似。僅調(diào)用靜態(tài) send() 方法,就能使用類的 缺省 版本:Transport.send(message);或者,讀者也可以從針對(duì)自己的協(xié)議的會(huì)話中獲得一個(gè)特定的實(shí)例,傳遞用戶名和密碼(如果不必要就不傳),發(fā)送消息,然后關(guān)閉連接。
·Javax.mail.Store:Store類實(shí)現(xiàn)特定郵件協(xié)議上的讀、寫(xiě)、監(jiān)視、查找等操作。通過(guò)Javax.mail.Store類可以訪問(wèn)Javax.mail.Folder類。
·Javax.mail.Folder:Folder類用于分級(jí)組織郵件,并提供照J(rèn)avax.mail.Message格式訪問(wèn)email的能力。
怎樣使用JavaScript/" target="_blank">JSP發(fā)送email?
下面我們通過(guò)一個(gè)簡(jiǎn)單的例子,來(lái)說(shuō)明在JavaScript/" target="_blank">JSP中郵件是如何發(fā)送的。該例由兩個(gè)文件組成。一個(gè)是HTML文件,用來(lái)建立郵件信息(包括發(fā)送人、接收人、主題等)的表單,并將表單內(nèi)容發(fā)送給JavaScript/" target="_blank">JSP文件;另外一個(gè)是JavaScript/" target="_blank">JSP頁(yè)面,負(fù)責(zé)郵件的發(fā)送。
HTML文件 <HTML> <BODY> <FORM action="sendmail.jsp" method="post"> <TABLE align="center"> <TR> <TD width="50%"> To:<BR><INPUT name="to" size="25"> </TD> <TD width="50%"> From:<BR><INPUT name="from" size="25"> </TD> </TR> <TR> <TD colspan="2"> Subject:<BR><INPUT name="subject" size="50"> </TD> </TR> <TR> <TD colspan="2"> <P>Message:<BR> <TEXTAREA name="text" rows=25 cols=85></TEXTAREA> </P> </TD> </TR> </TABLE> <INPUT type="submit" name="cb_submit" value=" Send "> <INPUT type="reset" name="cb_reset" value=" Clear "> </FORM> </BODY> </HTML> |
JavaScript/" target="_blank">JSP頁(yè)面的作用就是獲得表單提交過(guò)來(lái)的數(shù)據(jù),并將這些數(shù)據(jù)賦給Java Mail API中相應(yīng)的對(duì)象,最后完成郵件發(fā)送。
sendmail.jsp文件 <%@ page import=" Javax.mail.*, Javax.mail.internet.*, Javax.activation.*,Java.util.*"%> <html> <head> <TITLE>JavaScript/" target="_blank">JSP meets JavaMail, what a sweet combo.</TITLE> </head> <body> <% try{ Properties props = new Properties(); Session sendMailSession; Store store; Transport transport;
sendMailSession = Session.getInstance(props, null); props.put("mail.smtp.host", "smtp.jspinsider.com"); Message newMessage = new MimeMessage(sendMailSession); newMessage.setFrom(new InternetAddress(request.getParameter("from"))); newMessage.setRecipient(Message.RecipientType.TO, new InternetAddress ( request.getParameter ("to"))); newMessage.setSubject(request.getParameter("subject")); newMessage.setSentDate(new Date()); newMessage.setText(request.getParameter("text")); transport = sendMailSession.getTransport("smtp"); transport.send(newMessage); %> <P>Your mail has been sent.</P> <% }catch(MessagingException m) { out.println(m.toString()); } %> </body> </html> |
如何發(fā)送HTML類型的郵件
在上面的例子中,我們實(shí)現(xiàn)了如何發(fā)送文本格式的郵件,那么HTML格式的郵件該如何發(fā)送呢?那就看看下面的例子吧。
該例由四個(gè)文件組成:
·form.htm:用來(lái)建立郵件信息的表單
·send.jsp:用來(lái)獲取表單提交的信息,并調(diào)用mymail.mail.HTML.send()方法發(fā)送郵件
·StringDataSource.Java:用戶自定義的JavaBean,用來(lái)將郵件的Body部分轉(zhuǎn)化為HTML格式
·HTML.Java:用戶自定義的JavaBean,用來(lái)發(fā)送HTML格式的郵件。在send.jsp文件中提到的mymail.mail.HTML.send()方法就在該JavaBean中定義。
form.htm <html> <head> <meta http-equiv="Content-Type" content="text/html; charset=gb2312"> <title> JavaMail - 發(fā)送HTML郵件</title> </head> <body> <table border="0" cellspacing="0" cellpadding="0"> <form method="post" action="send.jsp"> <tr> <td>SMTP主機(jī):</td> <td><input type="text" name="smtp" size="80"></td> </tr> <tr> <td>發(fā)信人:</td> <td><input type="text" name="from" size="80"></td> </tr> <tr> <td>收信人:</td> <td><input type="text" name="to" size="80"></td> </tr> <tr> <td>抄送人:</td> <td><input type="text" name="cc" size="80"></td> </tr> <tr> <td>暗送人:</td> <td><input type="text" name="bcc" size="80"></td> </tr> <tr> <td>主題:</td> <td><input type="text" name="subject" size="80"></td> </tr> <tr> <td valign="top">內(nèi)容:</td> <td><textarea name="body" rows="5" cols="80"></textarea></td> </tr> <tr> <td colspan="2" align="center"><input type="submit" value="發(fā)送"></td> </tr> </form> </table> </body> </html>
send.jsp <% //變量聲明 Java.lang.String smtp,from,to,cc,bcc,subject,body;
//獲得用戶輸入數(shù)據(jù) smtp = request.getParameter("smtp"); from = request.getParameter("from"); to = request.getParameter("to"); cc = request.getParameter("cc"); bcc = request.getParameter("bcc"); subject = request.getParameter("subject"); if(subject!=null){ subject = new Java.lang.String(subject.getBytes("iso-8859-1")); } body = request.getParameter("body"); //發(fā)送郵件 mymail.mail.HTML.send(smtp,from,to,cc,bcc,subject,body); %>
mymail.jaf.StringDataSource.Java package mymail.jaf; public class StringDataSource implements Javax.activation.DataSource { private Java.lang.String data; private Java.lang.String type;
public StringDataSource(Java.lang.String data,Java.lang.String type){ this.data = data; this.type = type; }
public Java.io.InputStream getInputStream() throws Java.io.IOException{ return new Java.io.StringBufferInputStream(data); }
public Java.io.OutputStream getOutputStream() throws Java.io.IOException{ throw new Java.io.IOException("it does not support this method now!"); }
public Java.lang.String getContentType(){ return type; }
public Java.lang.String getName(){ return " mymail "; } }
mymail.mail.HTML.Java package mymail.mail;
public final class HTML { public static void send( Java.lang.String smtp, /*SMTP主機(jī)地址*/ Java.lang.String from, /*發(fā)信人*/ Java.lang.String to, /*收信人*/ Java.lang.String cc, /*抄送人*/ Java.lang.String bcc, /*暗送人*/ Java.lang.String subject, /*主題*/ Java.lang.String body /*內(nèi)容*/ ) throws Java.lang.Exception { //變量聲明 Java.util.Properties props; //系統(tǒng)屬性 Javax.mail.Session mailSession; //郵件會(huì)話對(duì)象 Javax.mail.internet.MimeMessage mimeMsg; //MIME郵件對(duì)象
//設(shè)置系統(tǒng)屬性 props = Java.lang.System.getProperties(); //獲得系統(tǒng)屬性對(duì)象 props.put("mail.smtp.host",smtp); //設(shè)置SMTP主機(jī)
//獲得郵件會(huì)話對(duì)象 mailSession = Javax.mail.Session.getDefaultInstance(props,null);
//創(chuàng)建MIME郵件對(duì)象 mimeMsg = new Javax.mail.internet.MimeMessage(mailSession);
//設(shè)置發(fā)信人 mimeMsg.setFrom(new Javax.mail.internet.InternetAddress(from));
//設(shè)置收信人 if(to!=null) { mimeMsg.setRecipients(Javax.mail.Message.RecipientType.TO,Javax.mail. internet.InternetAddress.parse(to)); }
//設(shè)置抄送人 if(cc!=null) { mimeMsg.setRecipients(Javax.mail.Message.RecipientType.CC,Javax.mail. internet.InternetAddress.parse(cc)); }
//設(shè)置暗送人 if(bcc!=null) { mimeMsg.setRecipients(Javax.mail.Message.RecipientType.BCC,Javax.mail. internet.InternetAddress.parse(bcc)); }
//設(shè)置郵件主題 //mimeMsg.setSubject(subject); mimeMsg.setSubject(subject,"gb2312");
//設(shè)置郵件內(nèi)容,將郵件body部分轉(zhuǎn)化為HTML格式 mimeMsg.setDataHandler(new Javax.activation.DataHandler(new mymail.jaf. StringDataSource (body,"text/html")));
//發(fā)送郵件 Javax.mail.Transport.send(mimeMsg); } } |
[page_break] 如何實(shí)現(xiàn)消息和標(biāo)志的刪除?
消息的刪除涉及到與消息相關(guān)的 Flags(標(biāo)志)。不同 flag 表示不同的狀態(tài),有些標(biāo)志由系統(tǒng)定義,而有些則由用戶自己定義。下面列出在內(nèi)部類 Flags.Flag 中預(yù)定義的標(biāo)志:
·Flags.Flag.ANSWERED
·Flags.Flag.DELETED
·Flags.Flag.DRAFT
·Flags.Flag.FLAGGED
·Flags.Flag.RECENT
·Flags.Flag.SEEN
·Flags.Flag.USER
上述這些標(biāo)志只是標(biāo)準(zhǔn)定義,并不意味著所有郵件服務(wù)器或供應(yīng)商都支持所有這些標(biāo)志。例如,除了刪除消息標(biāo)志外,POP 協(xié)議不再支持其它任何標(biāo)志。檢查是否存在新郵件,這不是個(gè) POP 任務(wù),而是內(nèi)建于郵件客戶機(jī)的任務(wù)。為找出哪些標(biāo)志能被支持,可以用 getPermanentFlags() 向 folder 提出請(qǐng)求。
要?jiǎng)h除消息,您可以設(shè)置消息的 DELETED flag:
message.setFlag(Flags.Flag.DELETED, true);
首先,請(qǐng)以 READ_WRITE 模式打開(kāi) folder:
folder.open(Folder.READ_WRITE);
然后,當(dāng)所有消息的處理完成后,關(guān)閉 folder,并傳遞一個(gè) true 值,從而擦除(expunge)有 delete 標(biāo)志的消息。
folder.close(true);
一個(gè) Folder 的 expunge() 方法可以用來(lái)刪除消息。但 Sun 的 POP3 供應(yīng)商不支持。其它供應(yīng)商有的或許能夠?qū)崿F(xiàn)這一功能,而有的則不能。IMAP 供應(yīng)商極有可能實(shí)現(xiàn)此功能。因?yàn)?POP 只支持單個(gè)對(duì)郵箱的訪問(wèn),對(duì) Sun 的供應(yīng)商來(lái)說(shuō),您必需關(guān)閉 folder 以刪除消息。
要取消標(biāo)志,只要傳遞 false 給 setFlag() 方法就行了。想知道是否設(shè)置過(guò)標(biāo)志,可以用 isSet() 檢查。
如何實(shí)現(xiàn)認(rèn)證?
想必讀者已經(jīng)知道 ,如果需要可以用一個(gè) Authenticator 提示用戶輸入用戶名和密碼,而不是將用戶名和密碼作為字符串傳遞。在這里您會(huì)明確了解怎樣更充分的使用Java Mail API的認(rèn)證機(jī)制。
不用主機(jī)、用戶名和密碼與 Store 相連接,而是設(shè)置 Properties 來(lái)?yè)碛兄鳈C(jī),然后告訴 Session 自定義的 Authenticator 實(shí)例,如下所示:
// Setup properties Properties props = System.getProperties(); props.put("mail.pop3.host", host);
// Setup authentication, get session Authenticator auth = new PopupAuthenticator(); Session session = Session.getDefaultInstance(props, auth);
// Get the store Store store = session.getStore("pop3"); store.connect(); |
然后,創(chuàng)建一個(gè) Authenticator 子類并從 getPasswordAuthentication() 方法中返回 PasswordAuthentication 對(duì)象。下面就是這樣一種實(shí)現(xiàn),其中用戶名和密碼僅占用一個(gè)域。(這不是一個(gè) Swing 工程教程;只要將兩部分輸入同一個(gè)域,用逗號(hào)分隔就行。)
import Javax.mail.*; import Javax.swing.*; import Java.util.*;
public class PopupAuthenticator extends Authenticator { public PasswordAuthentication getPasswordAuthentication() { String username, password; String result = JOptionPane.showInputDialog("Enter 'username,password'"); StringTokenizer st = new StringTokenizer(result, ","); username = st.nextToken(); password = st.nextToken(); return new PasswordAuthentication(username, password); } } |
因?yàn)?PopupAuthenticator 涉及到 Swing,它會(huì)啟動(dòng) AWT 的事件處理線程。這一點(diǎn)基本上要求您在代碼中添加一個(gè)對(duì) System.exit() 的調(diào)用來(lái)終止程序。
如何實(shí)現(xiàn)消息的轉(zhuǎn)發(fā)?
轉(zhuǎn)發(fā)消息,相對(duì)于其他功能來(lái)說(shuō),顯得有些棘手。沒(méi)有單獨(dú)的方法可以供調(diào)用,讀者必須通過(guò)對(duì)組成消息各部分的處理來(lái)組織要轉(zhuǎn)發(fā)的消息。
一條郵件消息可以由多個(gè)部分組成。在處理 MIME 消息時(shí),消息中每部分都是 BodyPart,再特殊些,是 MimeBodyPart。不同的 body part(信體部件或正文部件)結(jié)合成一個(gè)容器,名為 Multipart,再特殊些,就是 MimeMultipart。要轉(zhuǎn)發(fā)一條消息,您為自己的消息正文創(chuàng)建一個(gè)部件,要轉(zhuǎn)發(fā)的消息作為另一部件。并且將兩個(gè)部件結(jié)合成一個(gè) multipart(多部件)。然后您將這個(gè) multipart 添加到一則已寫(xiě)好恰當(dāng)?shù)刂返南⒅校l(fā)送。
本質(zhì)上就是如此。要將一條消息內(nèi)容復(fù)制到另一條,只要復(fù)制 DataHandler (JavaBeans Activation Framework 中的類)就行了。
// Create the message to forward Message forward = new MimeMessage(session); // Fill in header forward.setSubject("Fwd: " + message.getSubject()); forward.setFrom(new InternetAddress(from)); forward.addRecipient(Message.RecipientType.TO, new InternetAddress(to)); // Create your new message part BodyPart messageBodyPart = new MimeBodyPart(); messageBodyPart.setText( "Here you go with the original message:\n\n"); // Create a multi-part to combine the parts Multipart multipart = new MimeMultipart(); multipart.addBodyPart(messageBodyPart);
// Create and fill part for the forwarded content messageBodyPart = new MimeBodyPart(); messageBodyPart.setDataHandler(message.getDataHandler());
// Add part to multi part multipart.addBodyPart(messageBodyPart);
// Associate multi-part with message forward.setContent(multipart);
// Send message Transport.send(forward); |
怎樣處理附件?
附件是郵件消息的相關(guān)資源,如通常不包含在消息正文里文本文件、電子表格或圖像等。常見(jiàn)的郵件程序,如 Eudora 和 pine 之類,可以用 JavaMail API 將資源 attach(附加) 到您的消息上,就可以在收到消息時(shí)得到。
附件的發(fā)送:
發(fā)送附件非常像轉(zhuǎn)發(fā)消息。您建立各部分以組成完整消息。完成第一部件,即消息正文后,您添加其它部件,其中每個(gè) DataHandler 都代表附件,而不是轉(zhuǎn)發(fā)消息情況下的共享處理程序。如果從文件中讀附件,附件的數(shù)據(jù)源是 FileDataSource。而如果從 URL 中讀時(shí),附件的數(shù)據(jù)源是 URLDataSource。一旦存在 DataSource,只要先把它傳遞給 DataHandler 構(gòu)造器,最后再用 setDataHandler() 把它附加到 BodyPart。假定您要保留附件的原始文件名,最終要做的是用 BodyPart 的 setFileName() 方法設(shè)置與附件相關(guān)的文件名。如下所示:
// Define message Message message = new MimeMessage(session); message.setFrom(new InternetAddress(from)); message.addRecipient(Message.RecipientType.TO, new InternetAddress(to)); message.setSubject("Hello JavaMail Attachment");
// Create the message part BodyPart messageBodyPart = new MimeBodyPart();
// Fill the message messageBodyPart.setText("Pardon Ideas");
Multipart multipart = new MimeMultipart(); multipart.addBodyPart(messageBodyPart);
// Part two is attachment messageBodyPart = new MimeBodyPart(); DataSource source = new FileDataSource(filename); messageBodyPart.setDataHandler(new DataHandler(source)); messageBodyPart.setFileName(filename); multipart.addBodyPart(messageBodyPart);
// Put parts in message message.setContent(multipart);
// Send the message Transport.send(message); |
就消息引入附件時(shí),若程序是個(gè) servlet (小服務(wù)程序),除告知消息發(fā)送到何處外,還必需上載附件?梢詫 multipart/form-data 表單編碼類型(form encoding type)用于每個(gè)上載文件的處理。
<FORM ENCTYPE="multipart/form-data" method=post action="/myservlet"> 。糏NPUT TYPE="file" NAME="thefile"> 。糏NPUT TYPE="submit" VALUE="Upload"> </FORM> |
注意:消息大小由 SMTP 服務(wù)器而不是 JavaMail API 來(lái)限制。如果您碰到問(wèn)題,可以考慮用設(shè)置 ms 和 mx 參數(shù)的方法增大 Java 堆大小。
[page_break] 附件的獲。
從消息中獲取附件比發(fā)送它們棘手些,因?yàn)?MIME 沒(méi)有簡(jiǎn)單的關(guān)于附件的概念。當(dāng)消息包含附件時(shí),消息的內(nèi)容是個(gè) Multipart 對(duì)象。接著,您需要處理每個(gè) Part,獲取主要內(nèi)容和附件。標(biāo)有從 part.getDisposition() 獲得的 Part.ATTACHMENT 配置(disposition)的部件(Part)無(wú)疑就是附件。但是,沒(méi)有配置(以及一個(gè)非文本 MIME 類型)和帶 Part.INLINE 配置的部件也可能是附件。當(dāng)配置要么是 Part.ATTACHMENT,要么是 Part.INLINE 時(shí),這個(gè)消息部件的內(nèi)容就能被保存。只要用 getFileName() 和 getInputStream() 就能分別得到原始文件名和輸入流。
Multipart mp = (Multipart)message.getContent(); for (int i=0, n=multipart.getCount(); i<n; i++) { Part part = multipart.getBodyPart(i)); String disposition = part.getDisposition(); if ((disposition != null) && ((disposition.equals(Part.ATTACHMENT) (disposition.equals(Part.INLINE))) { saveFile(part.getFileName(), part.getInputStream()); } } |
saveFile() 方法僅依據(jù)文件名創(chuàng)建了一個(gè) File,它從輸入流中將字節(jié)讀出,然后寫(xiě)入到文件中。萬(wàn)一文件已經(jīng)存在,就在文件名后添加一個(gè)數(shù)字作為新文件名,如果這個(gè)文件名仍存在,則繼續(xù)添,直到找不到這樣的文件名為止。
// from saveFile() File file = new File(filename); for (int i=0; file.exists(); i++) { file = new File(filename+i); } |
上面的代碼涵蓋了最簡(jiǎn)單的情況 - 消息中各部件恰當(dāng)?shù)臉?biāo)記了。要涵蓋所有情況,還要在配置為空時(shí)進(jìn)行處理,并且獲取部件的 MIME 類型來(lái)進(jìn)行相應(yīng)處理。
if (disposition == null) { // Check if plain MimeBodyPart mbp = (MimeBodyPart)part; if (mbp.isMimeType("text/plain")) { // Handle plain } else { // Special non-attachment cases here of image/gif, text/html, ... } ... } |
如何一次發(fā)送多個(gè)郵件?
取得必要的Transport對(duì)象并調(diào)用SendMessage()發(fā)送每一個(gè)郵件。注意在調(diào)用之間設(shè)置或改變接收者。
Message message = …; Transport t = session.getTransport("smtp"); t.connect(); message.setRecipient(Message.RecipientType.TO,recipient1); t.sendMessage(message,recipient); message.setRecipient(Message.RecipientType.TO,recipient2); t.sendMessage(message,recipient); message.setRecipient(Message.RecipientType.TO,recipient3); t.sendMessage(message,recipient); t.close(); |
如何保存郵件?
用MimeMessage類的writeTo()方法可以實(shí)現(xiàn)。用Message類的對(duì)象不能實(shí)現(xiàn)。
FileOutputStream fos = new FileOutputStream("test.mail"); Mimemessage.writeTo(fos); |
怎樣發(fā)送帶有圖像的HTML格式郵件?
一些圖像還是放在服務(wù)器上較好,讓用戶的郵件工具處理。讀者可以將圖像作為附件或HTML體發(fā)送。如果所有附件保存在同一個(gè)目錄下,那么必須用不同的圖像文件名以確保郵件工具不會(huì)顯示其他圖片。另外圖像URL要用絕對(duì)路徑,不能用相對(duì)路徑。
如何設(shè)置∕獲取郵件的優(yōu)先級(jí)?
設(shè)置郵件的優(yōu)先級(jí),只需在消息頭中添加"X-Priority"屬性:
MimeMessage msg;
Msg.addHeader("X-Priority","1");
同樣道理,要獲取郵件的優(yōu)先級(jí)只要獲取"X-Priority"屬性的值就可以了:
String priority = msg.getHeader("X-Priority");
附錄:Java Mail API的基本概念
什么是Java Mail API
JavaMail API 是一個(gè)用于閱讀、編寫(xiě)和發(fā)送電子郵件的可選包(標(biāo)準(zhǔn)擴(kuò)展)。與 Eudora、pine 及 Microsoft Outlook 相似,這個(gè)包用來(lái)創(chuàng)建郵件用戶代理(Mail User Agent,MUA) 類型程序。API 的主要用途并不在于傳輸、發(fā)送和轉(zhuǎn)發(fā)消息;這一功能范圍屬于某些應(yīng)用程序,如 sendmail 及其它郵件傳輸代理(Mail Transfer Agent,MTA)類型程序。MUA 類型的程序能讓用戶閱讀和書(shū)寫(xiě)郵件,而它卻依賴 MTA 處理實(shí)際發(fā)送。
什么是SMTP
SMTP(Simple Mail Transfer Protocol),即簡(jiǎn)單郵件傳輸協(xié)議,它定義了發(fā)送電子郵件的機(jī)制。在 JavaMail API 環(huán)境中,基于 JavaMail 的程序?qū)⒑湍墓净蛞蛱鼐W(wǎng)服務(wù)供應(yīng)商的(Internet Service Provider's,ISP's)SMTP 服務(wù)器通信。SMTP 服務(wù)器可將消息中轉(zhuǎn)至接收方 SMTP 服務(wù)器,以便最終讓用戶經(jīng)由 POP 或 IMAP 獲得。這不是要求 SMTP 服務(wù)器成為開(kāi)放的中繼,盡管 SMTP 服務(wù)器支持身份驗(yàn)證,不過(guò)還是得確保它的配置正確。像配置服務(wù)器來(lái)中繼消息或添加刪除郵件賬號(hào)這類任務(wù)的實(shí)現(xiàn),JavaMail API 中并不支持。
什么是POP
POP(Post Office Protocol),即郵局協(xié)議。目前用的是版本3,所以人們通常將它稱為 POP3,RFC 1939 定義了這個(gè)協(xié)議。POP 和SMTP一樣,也是一種機(jī)制,Internet上大多數(shù)人通過(guò)它得到郵件。該協(xié)議規(guī)定每個(gè)用戶只能有一個(gè)郵箱的支持。這就是它所能做的,而這也造成了許多混淆。使用 POP 時(shí),用戶熟悉的許多性能并不是由 POP 協(xié)議支持的,如查看有幾封新郵件消息這一性能。這些性能內(nèi)建于如 Eudora 或 Microsoft Outlook 之類的程序中,它們能記住一些事,諸如最近一次收到的郵件,還能計(jì)算出有多少是新的。所以當(dāng)使用 JavaMail API 時(shí),如果讀者想要這類信息,就只能由自己來(lái)計(jì)算了。
什么是IMAP
IMAP(Internet Message Access Protocol),即Internet消息訪問(wèn)協(xié)議,是更高級(jí)的用于接收消息的協(xié)議,在 RFC 2060 中有它的定義。目前使用的IMAP版本為4,人們習(xí)慣將它稱為 IMAP4。在用到 IMAP 時(shí),郵件服務(wù)器必需支持這個(gè)協(xié)議。不能僅僅把使用 POP 的程序用于 IMAP,并指望它支持 IMAP 所有性能。假設(shè)郵件服務(wù)器支持 IMAP,基于 JavaMail 的程序可以利用這種情況--用戶在服務(wù)器上可以有多個(gè)文件夾(folder),并且這些文件夾可以被多個(gè)用戶共享。
因?yàn)橛羞@一更高級(jí)的性能,您也許會(huì)認(rèn)為所有用戶都會(huì)使用 IMAP。事實(shí)并不是這樣。要求服務(wù)器接收新消息,在用戶請(qǐng)求時(shí)發(fā)送到用戶手中,還要在每個(gè)用戶的多個(gè)文件夾中維護(hù)消息。這樣雖然能將消息集中備份,但隨著用戶長(zhǎng)期的郵件夾越來(lái)越大,到磁盤(pán)空間耗盡時(shí),每個(gè)用戶都會(huì)受到損失。使用 POP,就能卸載郵件服務(wù)器上保存的消息了。
什么是MIME
MIME(Multipurpose Internet Mail Extensions),即多用途Internet郵件擴(kuò)展標(biāo)準(zhǔn)。它不是郵件傳輸協(xié)議,而是對(duì)傳輸內(nèi)容的消息、附件及其它的內(nèi)容定義了格式。這里有很多不同的RFC(Requirement of Comment)文檔:RFC 822、RFC 2045、RFC 2046 和 RFC 2047。作為一個(gè) JavaMail API 的用戶,您通常不必對(duì)這些格式操心。無(wú)論如何,一定存在這些格式而且程序會(huì)用到它。