JavaMail:在Web应用下完整接收、解析复杂邮件

在上一篇文章 JavaMail:利用Tomcat和浏览器解析邮件内容 中只是简单地实现了把邮件正文部分的文本内容输出到浏览器中进行解析显示。当然,需要用到 Tomcat 服务器。

其实,让我们自己来解析一封复杂的邮件是很不容易的,邮件里面格式、规范复杂得很。幸运的是,我们所用的浏览器一定内置了解析各种数据类型的数据处理模块,我们只需要在把数据流传输给浏览器之前明确地指定该数据流属于哪种数据类型即可,之后一切的解析操作由浏览器自动帮我们完成。

现在我们想要像 Outlook 客户端或者登录网页邮箱那样子接收、查看邮件,而这些邮件中可能包含附件、正文中可能内嵌图片、音频等等。好吧,我们来实现一下吧。

程序结构:

1、mailboxLogin.html 文件用于提供一个用户登录界面,由用户填写 POP3 服务器主机名、用户名和密码;

2、connectServer.jsp 文件用于获取登录界面中的登录信息,并连接到 POP3 服务器,还要读取显示出邮箱中的所有邮件列表,再提供一个查看邮件详细内容的超链接;

3、showMails.jsp 文件用于将一个 Web 页面分成两帧,目的在上一篇文章中讲过,就是分别指定不同的 MIME 消息头类型来让浏览器解析、显示出邮件头内容和邮件正文内容;

4、ShowHeader.java 文件是处理邮件头内容的 Servlet 程序;

5、ShowContent.java 文件时处理邮件正文内容的 Servlet 程序;

6、HandleAttachments.java 文件是当从 ShowHeader.java 程序处理邮件头内容时,若发现该邮件包含附件,则交给 HandleAttachments.java 这个 Servlet 程序进行处理,它能够在邮件头中给出附件的文件名、超链接,所以用户可以点击下载。这是上篇文章中所缺乏的,也是这个程序实现中的重点!

7、web.xml 文件用于根据前面的 Servlet 程序来部署、配置相关的映射信息。

好吧,讲了那么多让人不知所云的东西,来点截图啊、代码啊最让我兴奋的了,因为截图、所有代码我保证都是完整的(或许自己尝试一下可以弄出更多的截图),嘻嘻…

我们依然需要 Tomcat 这个服务器,关于 JSP 、Servlet 编程技术、部署配置映射信息等方面的内容就不能多讲了。这里先说明,我们要登录的邮箱时 [email protected] ,用户名为 testhao ,密码为 123456,里面的邮件跟上篇文章重点截图一样,不过我自己多发送了一封包含3封附件的邮件哦,如下:(更多精彩截图,请继续看下文)

JavaMail:在Web应用下完整接收、解析复杂邮件_第1张图片

JavaMail:在Web应用下完整接收、解析复杂邮件_第2张图片

JavaMail:在Web应用下完整接收、解析复杂邮件_第3张图片

程序代码

mailboxLogin.html

 
 
  1. <hmtl>
  2. <head>
  3. <title>邮箱登录页面</title>
  4. </head>
  5. <body>
  6. <formaction="connectServer.jsp"method="post">
  7. POP3服务器主机名:<inputname="pop3Server"type="text"></br>
  8. 邮箱用户名:<inputname="user"type="text"></br>
  9. 邮箱密码:<inputname="pwd"type="password"></br>
  10. <inputtype="submit"value="登录">
  11. <inputtype="reset"value="重置"></br>
  12. </form>
  13. </body>
  14. </html>

connectServer.jsp

 
 
  1. <%@page
  2. import="javax.mail.*,java.util.Properties"
  3. contentType="text/html;charset=gbk"%>
  4. <%
  5. Stringpop3Server=request.getParameter("pop3Server");
  6. Stringuser=request.getParameter("user");
  7. Stringpwd=request.getParameter("pwd");
  8. //创建一个有具体连接信息的Properties对象
  9. Propertiesprops=newProperties();
  10. props.setProperty("mail.store.protocol","pop3");
  11. props.setProperty("mail.pop3.host",pop3Server);
  12. //使用Properties对象获得Session对象
  13. SessionmailSession=Session.getDefaultInstance(props);
  14. //mailSession.setDebug(true);
  15. Folderfolder=null;
  16. try{
  17. //利用Session对象获得Store对象,并连接pop3服务器
  18. Storestore=mailSession.getStore();
  19. store.connect(pop3Server,user,pwd);
  20. //获得邮箱内的邮件夹Folder对象,以"读-写"打开
  21. folder=store.getFolder("inbox");
  22. folder.open(Folder.READ_WRITE);
  23. }catch(Exceptione){
  24. e.printStackTrace();
  25. }
  26. //Folderfolder=POP3Connect.getFolder(pop3Server,user,pwd);
  27. session.setAttribute("folder",folder);
  28. Stringfrom=null;
  29. Stringsubject=null;
  30. Message[]messages=folder.getMessages();
  31. intmessageCounts=messages.length;
  32. for(inti=0;i<messageCounts;i++)
  33. {
  34. try{
  35. from=messages[i].getFrom()[0].toString();
  36. subject=messages[i].getSubject();
  37. out.println("<B>邮件"+(i+1)+"</B>");
  38. %>
  39. </br>发件人地址:<%=from%></br>
  40. 邮件主题:<%=subject%></br>
  41. <ahref="showMails.jsp?msgnum=<%=i+1%>">
  42. >>查看邮件!!!</a></br></br>
  43. <%
  44. }catch(Exceptione){
  45. e.printStackTrace();
  46. }
  47. }
  48. %>

showMails.jsp

 
 
  1. <framesetrows="25%,*">
  2. <framesrc="/JavaMail/ShowHeader?msgnum=
  3. <%=request.getParameter("msgnum")%>"scrolling="yes">
  4. <framesrc="/JavaMail/ShowContent?msgnum=
  5. <%=request.getParameter("msgnum")%>"scrolling="yes">
  6. </frameset>

ShowHeader.java

 
 
  1. importjava.io.*;
  2. importjava.text.DateFormat;
  3. importjavax.mail.internet.MimeUtility;
  4. importjavax.mail.*;
  5. importjavax.servlet.*;
  6. importjavax.servlet.http.*;
  7. //解析邮件头消息,包括判断是否有附件
  8. publicclassShowHeaderextendsHttpServlet
  9. {
  10. publicvoiddoGet(HttpServletRequestrequest,
  11. HttpServletResponseresponse)throwsServletException,IOException
  12. {
  13. response.setContentType("text/html;charset=gbk");
  14. PrintWriterout=response.getWriter();
  15. HttpSessionsession=request.getSession();
  16. Folderfolder=(Folder)session.getAttribute("folder");
  17. intmsgnum=Integer.parseInt(request.getParameter("msgnum"));
  18. try{
  19. Messagemessage=folder.getMessage(msgnum);
  20. Stringfrom=(message.getFrom()[0]).toString();
  21. Stringsubject=message.getSubject();
  22. StringsentDate=DateFormat.getInstance().format(message.getSentDate());
  23. out.println("邮件主题:"+subject+"<br/>");
  24. out.println("发件人地址:"+from+"<br/>");
  25. out.println("发送日期:"+sentDate+"<br/>");
  26. //如果该邮件是组合型"multipart/*"则可能包含附件等
  27. if(message.isMimeType("multipart/*"))
  28. {
  29. Multipartmultipart=(Multipart)message.getContent();
  30. intbodyCounts=multipart.getCount();
  31. for(inti=0;i<bodyCounts;i++)
  32. {
  33. BodyPartbodypart=multipart.getBodyPart(i);
  34. //如果该BodyPart对象包含附件,则应该解析出来
  35. if(bodypart.getDisposition()!=null)
  36. {
  37. Stringfilename=bodypart.getFileName();
  38. if(filename.startsWith("=?"))
  39. {
  40. //把文件名编码成符合RFC822规范
  41. filename=MimeUtility.decodeText(filename);
  42. }
  43. //生成打开附件的超链接
  44. out.print("<B>附件"+(i+1)+":</B>");
  45. out.println("<ahref=HandleAttachments?msgnum="
  46. +msgnum+"&&bodynum="+i+"&&filename="
  47. +filename+">"+filename+"</a></br>");
  48. }
  49. }
  50. }
  51. }catch(Exceptione){
  52. e.printStackTrace();
  53. }
  54. }
  55. }

ShowContent.java

 
 
  1. importjava.io.*;
  2. importjavax.mail.*;
  3. importjavax.servlet.*;
  4. importjavax.servlet.http.*;
  5. //处理邮件的正文部分
  6. publicclassShowContentextendsHttpServlet
  7. {
  8. publicvoiddoGet(HttpServletRequestrequest,
  9. HttpServletResponseresponse)throwsServletException,IOException
  10. {
  11. //获取输出流、Session会话对象、邮件夹Folder对象
  12. ServletOutputStreamout=response.getOutputStream();
  13. HttpSessionsession=request.getSession();
  14. Folderfolder=(Folder)session.getAttribute("folder");
  15. intmsgnum=Integer.parseInt(request.getParameter("msgnum"));
  16. try{
  17. Messagemessage=folder.getMessage(msgnum);
  18. if(!message.isMimeType("multipart/mixed"))
  19. {
  20. //若邮件类型不是"mixed"则表明不包含附件,
  21. //并设置类型让浏览器直接输出正文
  22. response.setContentType("message/rfc822");
  23. message.writeTo(out);
  24. }else{
  25. //如果是"mixed"型,则遍历所有BodyPart对象,
  26. //把不包含附件的邮件正文打印出来。
  27. //这是为了不让该程序既输出正文又传输附件大量的数据
  28. Multipartmultipart=(Multipart)message.getContent();
  29. intbodyCounts=multipart.getCount();
  30. for(inti=0;i<bodyCounts;i++)
  31. {
  32. BodyPartbodypart=multipart.getBodyPart(i);
  33. //不是"mixed"型且不包含附件
  34. if(!bodypart.isMimeType("multipart/mixed")
  35. &&bodypart.getDisposition()==null)
  36. {
  37. response.setContentType("message/rfc822");
  38. bodypart.writeTo(out);
  39. }
  40. }
  41. }
  42. }catch(Exceptione){
  43. e.printStackTrace();
  44. }
  45. }
  46. }

HandleAttachments.java

 
 
  1. importjava.io.*;
  2. importjavax.mail.*;
  3. importjavax.servlet.*;
  4. importjavax.servlet.http.*;
  5. //处理邮件中的附件
  6. publicclassHandleAttachmentsextendsHttpServlet
  7. {
  8. publicvoiddoGet(HttpServletRequestrequest,
  9. HttpServletResponseresponse)throwsServletException,IOException
  10. {
  11. response.setContentType("text/html;charset=gbk");
  12. PrintWriterout=response.getWriter();
  13. HttpSessionsession=request.getSession();
  14. intmsgnum=Integer.parseInt(request.getParameter("msgnum"));
  15. intbodynum=Integer.parseInt(request.getParameter("bodynum"));
  16. Folderfolder=(Folder)session.getAttribute("folder");
  17. Stringfilename=request.getParameter("filename");
  18. try{
  19. Messagemessage=folder.getMessage(msgnum);
  20. //将消息头类型设置为附件类型
  21. response.setHeader("Content-Disposition",
  22. "attachment;filename="+filename);
  23. Multipartmultipart=(Multipart)message.getContent();
  24. BodyPartbodypart=multipart.getBodyPart(bodynum);
  25. InputStreaminput=bodypart.getInputStream();
  26. inttemp=0;
  27. while((temp=input.read())!=-1)
  28. {
  29. out.write(temp);
  30. }
  31. }catch(Exceptione){
  32. e.printStackTrace();
  33. }
  34. }
  35. }

web.xml

 
 
  1. <?xmlversion="1.0"encoding="gb2312"?>
  2. <web-appxmlns="http://java.sun.com/xml/ns/j2ee"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  4. xsi:schemaLocation="http://java.sun.com/xml/ns/j2eehttp://java.sun.com/xml/ns/j2ee/web-app_2_4.xsd"
  5. version="2.4">
  6. <servlet>
  7. <servlet-name>ShowHeader</servlet-name>
  8. <servlet-class>ShowHeader</servlet-class>
  9. </servlet>
  10. <servlet-mapping>
  11. <servlet-name>ShowHeader</servlet-name>
  12. <url-pattern>/ShowHeader</url-pattern>
  13. </servlet-mapping>
  14. <servlet>
  15. <servlet-name>ShowContent</servlet-name>
  16. <servlet-class>ShowContent</servlet-class>
  17. </servlet>
  18. <servlet-mapping>
  19. <servlet-name>ShowContent</servlet-name>
  20. <url-pattern>/ShowContent</url-pattern>
  21. </servlet-mapping>
  22. <servlet>
  23. <servlet-name>HandleAttachments</servlet-name>
  24. <servlet-class>HandleAttachments</servlet-class>
  25. </servlet>
  26. <servlet-mapping>
  27. <servlet-name>HandleAttachments</servlet-name>
  28. <url-pattern>/HandleAttachments</url-pattern>
  29. </servlet-mapping>
  30. </web-app>

测试过程:(其实就是精彩的截图啦)

JavaMail:在Web应用下完整接收、解析复杂邮件_第4张图片

JavaMail:在Web应用下完整接收、解析复杂邮件_第5张图片

JavaMail:在Web应用下完整接收、解析复杂邮件_第6张图片

JavaMail:在Web应用下完整接收、解析复杂邮件_第7张图片

JavaMail:在Web应用下完整接收、解析复杂邮件_第8张图片

好了,精彩截图没了吗?不是的,我们还要与 126 邮箱中的解析、显示效果比对一下,如下:

JavaMail:在Web应用下完整接收、解析复杂邮件_第9张图片

又是这个问题:

为什么我们明明才上传了 3 个附件啊(见最上面的图),这里显示为 4 个了。我猜测的原因在前面的文章中也说过了,它把邮件正文中的图片也解析为附件了,不同的邮件服务器在解析发送、接收、解析邮件时都有自己的方式、策略,这个不理了。不过话说回来,我们自己写的程序好像更加合理一点哦,直接显示 3 个附件,真好!

小结:

1、在这个基于 Tomcat 的 Web 应用中,只用解析邮件中的附件是新的知识,其他的都讲过了;

2、关于程序代码结构、内容是否高效的问题,在这些文章中都没有太多注意,原因是比较注重实现想要的功能,这个不足要意识到并且逐步改过来;

3、关于 JavaMail 邮件开发的学习似乎就到此告一段落了,从最原始的用 Windows 下的 telnet 程序连接服务器手工一条一条命令地发送、接收邮件,到自己写了个 GUI 程序封装这些邮件发送命令以达到真正的纯文本邮件发送的功能,再到学习 JavaMail API 更轻松的创建邮件内容、邮件发送、邮件接收、运用 Tomcat 和浏览器实现邮件的解析…

噢,好像是学习完了。但是,编程学习要达到卖油翁的“无他, 唯手熟尔”绝对是一条漫长艰苦的道路,在此源于大家互相交流、共同学习、一起进步!加油!!!

其实,有时间的话我们也可以扩展上面的程序,综合前面所有文章中的内容,实现一个能够发送邮件,上传图片、附件,以及接收、解析、查看邮件的综合应用…


你可能感兴趣的:(javamail)