a我考网

 找回密码
 立即注册

QQ登录

只需一步,快速开始

扫一扫,访问微社区

查看: 292|回复: 6

[JAVA] 2011年计算机等考二级JAVA学习精华整理(77)

[复制链接]
发表于 2012-7-31 22:04:26 | 显示全部楼层 |阅读模式
3.5 JavaMail(JAVA邮件服务)API详解  一、JavaMail API简介
, n0 {& Z! U' `# F  JavaMail API是读取、撰写、发送电子信息的可选包。我们可用它来建立如Eudora、Foxmail、MS Outlook Express一般的邮件用户代理程序(Mail User Agent,简称MUA)。而不是像sendmail或者其它的邮件传输代理(Mail Transfer Agent,简称MTA)程序那样可以传送、递送、转发邮件。从另外一个角度来看,我们这些电子邮件用户日常用MUA程序来读写邮件,而MUA依赖着MTA处理邮件的递送。  D  O3 T! d9 |! S4 Q3 G
  在清楚了到MUA与MTA之间的关系后,让我们看看JavaMail API是如何提供信息访问功能的吧!JavaMail API被设计用于以不依赖协议的方式去发送和接收电子信息,这个API被分为两大部分:. h6 D  F& I' Y1 d9 m% v  Q2 e
  基本功能:如何以不依赖于协议的方式发送接收电子信息,这也是本文所要描述的,不过在下文中,大家将看到这只是一厢情愿而已。
4 l" L- \% X6 l- Q  第二个部分则是依赖特定协议的,比如SMTP、POP、IMAP、NNTP协议。在这部分的JavaMail API是为了和服务器通讯,并不在本文的内容中。
6 l/ ]1 c9 }1 }$ |7 R: P1 L1 u  二、相关协议一览
& _* |$ j4 Y% L4 ?' |  在我们步入JavaMail API之前,先看一下API所涉及的协议。以下便是大家日常所知、所乐于使用的4大信息传输协议:' u/ C/ d+ C1 M) t( M/ j  y+ _- z2 H
  SMTP( [- u4 t0 a- ?' a. C+ M$ w+ y9 q
  POP
3 K5 E, Q# g" o# m7 P  IMAP1 q- C( X$ j- ^$ x2 m9 G7 f- Y
  MIME
/ P5 a2 g) M; b/ m5 a" N3 x' a  当然,上面的4个协议,并不是全部,还有NNTP和其它一些协议可用于传输信息,但是由于不常用到,所以本文便不提及了。理解这4个基本的协议有助于我们更好的使用JavaMail API。然而JavaMail API是被设计为与协议无关的,目前我们并不能克服这些协议的束缚。确切的说,如果我们使用的功能并不被我们选择的协议支持,那么JavaMail API并不可能如魔术师一样神奇的赋予我们这种能力。
) f, m" _' u1 r8 H% @: g" j  1.SMTP4 i# q  I& J5 w- a) ~
  简单邮件传输协议定义了递送邮件的机制。在下文中,我们将使用基于Java-Mail的程序与公司或者ISP的SMTP服务器进行通讯。这个SMTP服务器将邮件转发到接收者的SMTP服务器,直至最后被接收者通过POP或者IMAP协议获取。这并不需要SMTP服务器使用支持授权的邮件转发,但是却的确要注意SMTP服务器的正确设置(SMTP服务器的设置与JavaMail API无关)。$ g; s6 g$ u0 R( {+ c1 ^
  2.POP
$ g: w) l) {$ p$ m  }' D  POP是一种邮局协议,目前为第3个版本,即众所周知的POP3。POP定义了一种用户如何获得邮件的机制。它规定了每个用户使用一个单独的邮箱。大多数人在使用POP时所熟悉的功能并非都被支持,例如查看邮箱中的新邮件数量。而这个功能是微软的Outlook内建的,那么就说明微软Outlook之类的邮件客户端软件是通过查询最近收到的邮件来计算新邮件的数量来实现前面所说的功能。因此在我们使用JavaMail API时需要注意,当需要获得如前面所讲的新邮件数量之类的信息时,我们不得不自己进行计算。. Q! t% B  `% C
  3.IMAP
  k0 p. s, d; C) B8 S7 w+ C  IMAP使用在接收信息的高级协议,目前版本为第4版,所以也被称为IMAP4。需要注意的是在使用IMAP时,邮件服务器必须支持该协议。从这个方面讲,我们并不能完全使用IMAP来替代POP,不能期待IMAP在任何地方都被支持。假如邮件服务器支持IMAP,那么我们的邮件程序将能够具有以下被IMAP所支持的特性:每个用户在服务器上可具有多个目录,这些目录能在多个用户之间共享。. }+ A) y: V0 G, Q$ X
  其与POP相比高级之处显而易见,但是在尝试采取IMAP时,我们认识到它并不是十分完美的:由于IMAP需要从其它服务器上接收新信息,将这些信息递送给用户,维护每个用户的多个目录,这都为邮件服务器带来了高负载。并且IMAP与POP的一个不同之处是POP用户在接收邮件时将从邮件服务器上下载邮件,而IMAP允许用户直接访问邮件目录,所以在邮件服务器进行备份作业时,由于每个长期使用此邮件系统的用户所用的邮件目录会占有很大的空间,这将直接导致邮件服务器上磁盘空间暴涨。
回复

使用道具 举报

 楼主| 发表于 2012-7-31 22:04:27 | 显示全部楼层

2011年计算机等考二级JAVA学习精华整理(77)

 4.MIME  MIME并不是用于传送邮件的协议,它作为多用途邮件的扩展定义了邮件内容的格式:信息格式、附件格式等等。一些RFC标准都涉及了MIME:RFC 822, RFC 2045, RFC 2046, RFC 2047,有兴趣的Matrixer可以阅读一下。而作为JavaMail API的开发者,我们并不需关心这些格式定义,但是这些格式被用在了程序中。
4 W! T+ q, f" I' s: v& C  5.NNTP和其它的第三方协议
- Z6 t- c2 d) ?, F9 K  正因为JavaMail API在设计时考虑到与第三方协议实现提供商之间的分离,故我们可以很容易的添加一些第三方协议。SUN维护着一个第三方协议实现提供商的列表:http://java.sun.com/products/javamail/Third_Party.html,通过此列表我们可以找到所需要的而又不被SUN提供支持的第三方协议:比如NNTP这个新闻组协议和S/MIME这个安全的MIME协议。! W9 d" k% ]0 e' q
  三、安装7 k& n1 H7 A- u: q0 ?( ]* d5 h
  1.安装JavaMail% L6 M8 l7 D! y! s2 ]- q
  为了使用JavaMail API,需要从http://java.sun.com/products/javamail/downloads/index.html下载文件名格式为javamail-[version].zip的文件(这个文件中包括了JavaMail实现),并将其中的mail.jar文件添加到CLASSPATH中。这个实现提供了对SMTP、IMAP4、POP3的支持。
1 u+ t- Y# x$ P1 r; v  注意:在安装JavaMail实现之后,我们将在demo目录中发现许多有趣的简单实例程序。% m! y& m' Z% Y  I) O6 p- q
  在安装了JavaMail之后,我们还需要安装JavaBeans Activation Framework,因为这个框架是JavaMail API所需要的。如果我们使用J2EE的话,那么我们并无需单独下载JavaMail,因为它存在于J2EE.jar中,只需将J2EE.jar加入到CLASSPATH即可。2 s  Z: |; p4 [3 X- X+ W3 L
  2.安装JavaBeans Activation Framework$ Q2 o' U0 c& \# b1 c" {0 ]
  从http://java.sun.com/products/javabeans/glasgow/jaf.html下载JavaBeans Activation Framework,并将其添加到CLASSPATH中。此框架增加了对任何数据块的分类、以及对它们的处理的特性。这些特性是JavaMail API需要的。虽然听起来这些特性非常模糊,但是它对于我们的JavaMail API来说只是提供了基本的MIME类型支持。
! G6 E% M+ {* ^3 G# U$ F! c  到此为止,我们应当把mail.jar和activation.jar都添加到了CLASSPATH中。+ t2 D( \9 S7 r1 P$ U
  当然如果从方便的角度讲,直接把这两个Jar文件复制到JRE目录的lib/ext目录中也可以。
回复 支持 反对

使用道具 举报

 楼主| 发表于 2012-7-31 22:04:28 | 显示全部楼层

2011年计算机等考二级JAVA学习精华整理(77)

 四、初次认识JavaMail API  1.了解我们的JavaMail环境
6 X: U" {; Z' j& F+ J" C" v  A.纵览JavaMail核心类结构
; p2 }7 R7 k; Y  打开JavaMail.jar文件,我们将发现在javax.mail的包下面存在着一些核心类:Session、Message、Address、Authenticator、Transport、Store、Folder。而且在javax.mail.internet包中还有一些常用的子类。
% e1 O  s! j# x- ?. i% p0 [  B.Session9 |; m) y8 _8 R% B
  Session类定义了基本的邮件会话。就像Http会话那样,我们进行收发邮件的工作都是基于这个会话的。Session对象利用了java.util.Properties对象获得了邮件服务器、用户名、密码信息和整个应用程序都要使用到的共享信息。- }( m2 |9 d# a: S" c" F
  Session类的构造方法是私有的,所以我们可以使用Session类提供的getDefaultInstance()这个静态工厂方法获得一个默认的Session对象:
- z0 b3 S' O6 l# Q, m+ u  Properties props = new Properties();// fill props with any informationSession session = Session.getDefaultInstance(props, null);
+ W; J6 c3 z' U( H  或者使用getInstance()这个静态工厂方法获得自定义的Session:" F/ r4 @" u/ i) i3 W- ]
  Properties props = new Properties();// fill props with any informationSession session = Session.getInstance(props, null);
" L4 |' X0 }' {- D4 m1 t  从上面的两个例子中不难发现,getDefaultInstance()和getInstance()方法的第二个参数都是null,这是因为在上面的例子中并没有使用到邮件授权,下文中将对授权进行详细介绍。
7 e& F, D- z' \  从很多的实例看,在对mail server进行访问的过程中使用共享的Session是足够的,即使是工作在多个用户邮箱的模式下也不例外。! m4 s$ H! r1 d0 l( [7 C& [* @1 L
  C.Message% Z2 W+ ?+ M+ U3 I2 f) @
  当我们建立了Session对象后,便可以被发送的构造信息体了。在这里SUN提供了Message类型来帮助开发者完成这项工作。由于Message是一个抽象类,大多数情况下,我们使用javax.mail.internet.MimeMessage这个子类,该类是使用MIME类型、MIME信息头的邮箱信息。信息头只能使用US-ASCII字符,而非ASCII字符将通过编码转换为ASCII的方式使用。5 t9 x5 ?5 e/ w
  为了建立一个MimeMessage对象,我们必须将Session对象作为MimeMessage构造方法的参数传入:
$ a0 D3 ~. E5 C1 @5 d8 q% i- w2 X  MimeMessage message = new MimeMessage(session);2 J6 I' g  c- \! K
  注意:对于MimeMessage类来讲存在着多种构造方法,比如使用输入流作为参数的构造方法。
9 q- R+ L+ n$ F3 \  在建立了MimeMessage对象后,我们需要设置它的各个part,对于MimeMessage类来说,这些part就是MimePart接口。最基本的设置信息内容的方法就是通过表示信息内容和米么类型的参数调用setContent()方法:
1 w" Q; ]9 O; N; ^. X8 ?  message.setContent("Hello", "text/plain");& _/ [9 K+ J7 R. g) J  Y
  然而,如果我们所使用的MimeMessage中信息内容是文本的话,我们便可以直接使用setText()方法来方便的设置文本内容。
4 f; F0 z1 p2 ^" Q: I) B  message.setText("Hello");
; d. u# r& l; Q. c  m) C0 i) g  前面所讲的两种方法,对于文本信息,后者更为合适。而对于其它的一些信息类型,比如HTML信息,则要使用前者。
" p" M' n3 [9 h4 a' O" T  别忘记了,使用setSubject()方法对邮件设置邮件主题:  d' s5 @; L- n% p
  message.setSubject("First");+ h* `3 c4 }5 `
  D.Address
# k; a1 [' e( n  到这里,我们已经建立了Session和Message,下面将介绍如何使用邮件地址类:Address。像Message一样,Address类也是一个抽象类,所以我们将使用javax.mail.internet.InternetAddress这个子类。
4 P5 E3 g5 j' K3 \3 l# x( ?  通过传入代表邮件地址的字符串,我们可以建立一个邮件地址类:8 T4 ?- F: G: V# l" a" ]8 _" [: [. H
  Address address = new InternetAddress("president@whitehouse.gov");
回复 支持 反对

使用道具 举报

 楼主| 发表于 2012-7-31 22:04:29 | 显示全部楼层

2011年计算机等考二级JAVA学习精华整理(77)

  如果要在邮件地址后面增加名字的话,可以通过传递两个参数:代表邮件地址和名字的字符串来建立一个具有邮件地址和名字的邮件地址类:
# F( R) G1 a7 k, Q4 }  Address address = new InternetAddress("president@whitehouse.gov", "George Bush");' N" y5 u' @, Y* D. g- w, z3 f4 e
  本文在这里所讲的邮件地址类是为了设置邮件信息的发信人和收信人而准备的,在建立了邮件地址类后,我们通过message的setFrom()和setReplyTo()两种方法设置邮件的发信人:
6 D4 f- s4 S" g! O) j  message.setFrom(address);message.setReplyTo(address);
4 k- Y2 [& M/ m0 g$ B2 i! @) v  若在邮件中存在多个发信人地址,我们可用addForm()方法增加发信人:
) S6 p; |  @7 |  Address address[] = ...;message.addFrom(address);
! d/ A4 }" y8 j( L  为了设置收信人,我们使用addRecipient()方法增加收信人,此方法需要使用Message.RecipientType的常量来区分收信人的类型:
+ {% u4 Q5 r& A  message.addRecipient(type, address): _' G) {+ X! T4 |
  下面是Message.RecipientType的三个常量:' F( d- H+ }( `) o2 ~
  Message.RecipientType.TO
2 }& X7 H# @' G, w/ ?0 U  Message.RecipientType.CC
) q2 N9 d# _, t2 r  Message.RecipientType.BCC0 q# H3 `4 X, V4 O5 [# A
  因此,如果我们要发送邮件给总统,并发用一个副本给第一夫人的话,下面的方法将被用到:
" v, X8 Z5 @% S2 _  Address toAddress = new InternetAddress("vice.president@whitehouse.gov");Address ccAddress = new InternetAddress("first.lady@whitehouse.gov");message.addRecipient(Message.RecipientType.TO, toAddress);message.addRecipient(Message.RecipientType.CC, ccAddress);! P9 X; T# c& Q# y
  JavaMail API并没有提供检查邮件地址有效性的机制。当然我们可以自己完成这个功能:验证邮件地址的字符是否按照RFC822规定的格式书写或者通过DNS服务器上的MX记录验证等。- z# D* R6 a$ D" [" t
  E.Authenticator# U* a: P. E, R& A' ]& F/ ~8 N
  像java.net类那样,JavaMail API通过使用授权者类(Authenticator)以用户名、密码的方式访问那些受到保护的资源,在这里“资源”就是指邮件服务器。在javax.mail包中可以找到这个JavaMail的授权者类(Authenticator)。
1 Z, e: b9 H1 B3 L  在使用Authenticator这个抽象类时,我们必须采用继承该抽象类的方式,并且该继承类必须具有返回PasswordAuthentication对象(用于存储认证时要用到的用户名、密码)getPasswordAuthentication()方法。并且要在Session中进行注册,使Session能够了解在认证时该使用哪个类。
+ K) l( J6 A( W. g  下面代码片断中的MyAuthenticator就是一个Authenticator的子类。
( q; Y$ P3 L- O" B  Properties props = new Properties();// fill props with any informationAuthenticator auth = new MyAuthenticator();Session session = Session.getDefaultInstance(props, auth);1 |3 p$ B8 q% r0 o( x; k
  F.Transport
  {% o; L5 b3 T$ h6 B! D  在发送信息时,Transport类将被用到。这个类实现了发送信息的协议(通称为SMTP),此类是一个抽象类,我们可以使用这个类的静态方法send()来发送消息:* t8 }' E, y- c
  Transport.send(message);
回复 支持 反对

使用道具 举报

 楼主| 发表于 2012-7-31 22:04:30 | 显示全部楼层

2011年计算机等考二级JAVA学习精华整理(77)

 当然,方法是多样的。我们也可由Session获得相应协议对应的Transport实例。并通过传递用户名、密码、邮件服务器主机名等参数建立与邮件服务器的连接,并使用sendMessage()方法将信息发送,最后关闭连接:  message.saveChanges(); // implicit with send()Transport transport = session.getTransport("smtp");transport.connect(host, username, password);transport.sendMessage(message, message.getAllRecipients());transport.close();# P! Q- Z3 @" u5 J5 o0 G5 U$ v* e
  评论:上面的方法是一个很好的方法,尤其是在我们在同一个邮件服务器上发送多个邮件时。因为这时我们将在连接邮件服务器后连续发送邮件,然后再关闭掉连接。send()这个基本的方法是在每次调用时进行与邮件服务器的连接的,对于在同一个邮件服务器上发送多个邮件来讲可谓低效的方式。
1 {1 V$ t/ U$ f  注意:如果需要在发送邮件过程中监控mail命令的话,可以在发送前设置debug标志:4 i. s' _2 f/ {+ o; c
  session.setDebug(true)。
* O; Q+ W! |6 i$ P  G.Store和Folder
2 z3 X6 d7 X! z4 }6 I# K5 S  接收邮件和发送邮件很类似都要用到Session。但是在获得Session后,我们需要从Session中获取特定类型的Store,然后连接到Store,这里的Store代表了存储邮件的邮件服务器。在连接Store的过程中,极有可能需要用到用户名、密码或者Authenticator。* e* w7 m9 V" e) h5 D
  // Store store = session.getStore("imap");Store store = session.getStore("pop3");store.connect(host, username, password);# ^% p" M% u& r# q
  在连接到Store后,一个Folder对象即目录对象将通过Store的getFolder()方法被返回,我们可从这个Folder中读取邮件信息:5 C; Z( I6 S* e0 x; q# T( `
  Folder folder = store.getFolder("INBOX");folder.open(Folder.READ_ONLY);Message message[] = folder.getMessages();  t7 u! d+ L: g; Q" J2 P
  上面的例子首先从Store中获得INBOX这个Folder(对于POP3协议只有一个名为INBOX的Folder有效),然后以只读(Folder.READ_ONLY)的方式打开Folder,最后调用Folder的getMessages()方法得到目录中所有Message的数组。
+ K- A7 l* B+ G. t  注意:对于POP3协议只有一个名为INBOX的Folder有效,而对于IMAP协议,我们可以访问多个Folder(想想前面讲的IMAP协议)。而且SUN在设计Folder的getMessages()方法时采取了很智能的方式:首先接收新邮件列表,然后再需要的时候(比如读取邮件内容)才从邮件服务器读取邮件内容。2 X4 V  w5 t% ?9 l
  在读取邮件时,我们可以用Message类的getContent()方法接收邮件或是writeTo()方法将邮件保存,getContent()方法只接收邮件内容(不包含邮件头),而writeTo()方法将包括邮件头。' Y0 J1 p: }1 p3 Y" V  h
  System.out.println(((MimeMessage)message).getContent());: {  k3 }! r0 V7 Q- V/ p  @% Q$ I
  在读取邮件内容后,别忘记了关闭Folder和Store。" [8 n; x5 G* s9 l/ ]! o& k
  folder.close(aBoolean);store.close();- N, j3 C  L4 Y% d! R
  传递给Folder.close()方法的boolean 类型参数表示是否在删除操作邮件后更新Folder。# ^# _1 ~  p9 E" L7 O& |
  H.继续向前进!
  d" S4 O. G" ]  s6 M% k. B) i1 [  在讲解了以上的七个Java Mail核心类定义和理解了简单的代码片断后,下文将详细讲解怎样使用这些类实现JavaMail API所要完成的高级功能。
回复 支持 反对

使用道具 举报

 楼主| 发表于 2012-7-31 22:04:31 | 显示全部楼层

2011年计算机等考二级JAVA学习精华整理(77)

五、使用JavaMail API  在明确了JavaMail API的核心部分如何工作后,本人将带领大家学习一些使用Java Mail API任务案例。
8 X1 r  X0 v7 x: G, D/ R  1.发送邮件
9 \, |  n7 {6 M) r+ e1 h5 c+ L  在获得了Session后,建立并填入邮件信息,然后发送它到邮件服务器。这便是使用Java Mail API发送邮件的过程,在发送邮件之前,我们需要设置SMTP服务器:通过设置Properties的mail.smtp.host属性。) ~8 b4 r% Z0 ^
  String host = ...;String from = ...;String to = ...;// Get system propertiesProperties props = System.getProperties();// Setup mail serverprops.put("mail.smtp.host", host);// Get sessionSession session = Session.getDefaultInstance(props, null);// Define messageMimeMessage message = new MimeMessage(session);message.setFrom(new InternetAddress(from));message.addRecipient(Message.RecipientType.TO, new InternetAddress(to));message.setSubject("Hello JavaMail");message.setText("Welcome to JavaMail");// Send messageTransport.send(message);9 _% ~) U- A/ D. H2 v$ o! ~
  由于建立邮件信息和发送邮件的过程中可能会抛出异常,所以我们需要将上面的代码放入到try-catch结构块中。
" g/ e! b7 `1 O2 ?' C4 j2 s1 g; U  2.接收邮件5 S5 w& W1 i9 g3 j
  为了在读取邮件,我们获得了session,并且连接到了邮箱的相应store,打开相应的Folder,然后得到我们想要的邮件,当然别忘记了在结束时关闭连接。0 f% f  F2 ]- l/ g" P+ D
  String host = ...;String username = ...;String password = ...;// Create empty propertiesProperties props = new Properties();// Get sessionSession session = Session.getDefaultInstance(props, null);// Get the storeStore store = session.getStore("pop3");store.connect(host, username, password);// Get folderFolder folder = store.getFolder("INBOX");folder.open(Folder.READ_ONLY);// Get directoryMessage message[] = folder.getMessages();for (int i=0, n=message.length; i
$ d8 |2 b0 C6 g% I! i4 q  上面的代码所作的是从邮箱中读取每个邮件,并且显示邮件的发信人地址和主题。从技术角度讲,这里存在着一个异常的可能:当发信人地址为空时,getFrom()[0]将抛出异常。
7 u6 ]+ |$ Z& I9 q* @  下面的代码片断有效的说明了如何读取邮件内容,在显示每个邮件发信人和主题后,将出现用户提示从而得到用户是否读取该邮件的确认,如果输入YES的话,我们可用Message.writeTo(java.io.OutputStream os)方法将邮件内容输出到控制台上,关于Message.writeTo()的具体用法请看JavaMail API。8 C* d- w7 ?! A$ z6 v
  BufferedReader reader = new BufferedReader ( new InputStreamReader(System.in));// Get directoryMessage message[] = folder.getMessages();for (int i=0, n=message.length; i
! i9 Y, ~; G, t& `6 i, [1 x* E  窗体底端( N3 e& R: y" a- E9 O
  B.读取邮件中的附件. {+ J) f3 S6 T2 p% T- V& T
  读取邮件中的附件的过程要比发送它的过程复杂一点。因为带有附件的邮件是多部分组成的,我们必须处理每一个部分获得邮件的内容和附件。
: _( O1 A8 r- f% T9 u8 O  但是如何辨别邮件信息内容和附件呢?Sun在Part类(BodyPart类实现的接口类)中提供了getDisposition()方法让开发者获得邮件体部分的部署类型,当该部分是附件时,其返回之将是Part.ATTACHMENT。但附件也可以没有部署类型的方式存在或者部署类型为Part.INLINE,无论部署类型为Part.ATTACHMENT还是Part.INLINE,我们都能把该邮件体部分导出保存。
  m$ }1 O; K# o! B( _& @# Q1 }  Multipart mp = (Multipart)message.getContent();for (int i=0, n=multipart.getCount(); i9 [# m/ X  [, @7 }$ I* z: V
  下列代码中使用了saveFile方法是自定义的方法,它根据附件的文件名建立一个文件,如果本地磁盘上存在名为附件的文件,那么将在文件名后增加数字表示区别。然后从邮件体中读取数据写入到本地文件中(代码省略)。6 T3 Q2 w' X' ~, w. h7 B: P& a+ D
  // from saveFile()File file = new File(filename);for (int i=0; file.exists(); i++) { file = new File(filename+i);}( z$ N3 V" c  `
  以上是邮件体部分被正确设置的简单例子,如果邮件体部分的部署类型为null,那么我们通过获得邮件体部分的MIME类型来判断其类型作相应的处理,代码结构框架如下:1 Q$ g4 d. H7 g5 _/ g6 x# S
  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, ... }...}
0 [; J' ~7 e: f: l  8.处理HTML邮件
8 ?- y' X* G+ k" s  前面的例子中发送的邮件都是以文本为内容的(除了附件),下面将介绍如何接收和发送基于HTML的邮件。
% U) Y. V3 d  O2 B+ i$ i! Y7 L  A.发送HTML邮件2 u- t6 C4 e& e2 E7 S: _2 a
  假如我们需要发送一个HTML文件作为邮件内容,并使邮件客户端在读取邮件时获取相关的图片或者文字的话,只要设置邮件内容为html代码,并设置内容类型为text/html即可:
2 X3 D1 ^9 [9 U" U* x- [  String htmlText = "! p, g8 k6 F! ]
  Hello8 J' ]8 @5 Z9 ~$ E4 @
  " + "";message.setContent(htmlText, "text/html"));
- V$ u7 _2 i/ r4 T! q+ [0 M  请注意:这里的图片并不是在邮件中内嵌的,而是在URL中定义的。邮件接收者只有在线时才能看到。, `; l8 C5 A4 Q: d, t$ v
  在接收邮件时,如果我们使用JavaMail API接收邮件的话是无法实现以HTML方式显示邮件内容的。因为JavaMail API邮件内容视为二进制流。所以要显示HTML内容的邮件,我们必须使用JEditorPane或者第三方HTML展现组件。
2 Q% T# W# X- r- ]' r% [  以下代码显示了如何使用JEditorPane显示邮件内容:5 a  N; W* c6 f' y  H
  if (message.getContentType().equals("text/html")) { String content = (String)message.getContent(); JFrame frame = new JFrame(); JEditorPane text = new JEditorPane("text/html", content); text.setEditable(false); JScrollPane pane = new JScrollPane(text); frame.getContentPane().add(pane); frame.setSize(300, 300); frame.setDefaultCloseOperation(JFrame.DISPOSE_ON_CLOSE); frame.show();}
回复 支持 反对

使用道具 举报

 楼主| 发表于 2012-7-31 22:04:32 | 显示全部楼层

2011年计算机等考二级JAVA学习精华整理(77)

 B.在邮件中包含图片  如果我们在邮件中使用HTML作为内容,那么最好将HTML中使用的图片作为邮件的一部分,这样无论是否在线都会正确的显示HTML中的图片。处理方法就是将HTML中用到的图片作为邮件附件并使用特殊的cid URL作为图片的引用,这个cid就是对图片附件的Content-ID头的引用。
+ b& d" c- Q, j2 f! ^  处理内嵌图片就像向邮件中添加附件一样,不同之处在于我们必须通过设置图片附件所在的邮件体部分的header中Content-ID为一个随机字符串,并在HTML中img的src标记中设置为该字符串。这样就完成了图片附件与HTML的关联。
$ ]8 `( I5 ~' Q: I/ r. ?) I  String file = ...;// Create the messageMessage message = new MimeMessage(session);// Fill its headersmessage.setSubject("Embedded Image");message.setFrom(new InternetAddress(from));message.addRecipient(Message.RecipientType.TO, new InternetAddress(to));// Create your new message partBodyPart messageBodyPart = new MimeBodyPart();String htmlText = "9 x, m# ^) g6 B4 g5 H
  Hello' F  o/ J  {- Z) F: o, u& ?0 J
  " + "";messageBodyPart.setContent(htmlText, "text/html");// Create a related multi-part to combine the partsMimeMultipart multipart = new MimeMultipart("related");multipart.addBodyPart(messageBodyPart);// Create part for the imagemessageBodyPart = new MimeBodyPart();// Fetch the image and associate to partDataSource fds = new FileDataSource(file);messageBodyPart.setDataHandler(new DataHandler(fds));messageBodyPart.setHeader("Content-ID","");// Add part to multi-partmultipart.addBodyPart(messageBodyPart);// Associate multi-part with messagemessage.setContent(multipart);
! z9 [4 M! G3 y  9.在邮件中搜索短语
! q. h) F9 a4 b$ Y  JavaMail API提供了过滤器机制,它被用来建立搜索短语。这个短语由javax.mail.search包中的SearchTerm抽象类来定义,在定义后我们便可以使用Folder的Search()方法在Folder中查找邮件:
, ~9 \2 g; E4 E( y5 I  SearchTerm st = ...;Message[] msgs = folder.search(st);
, S' `" t8 `7 @. |5 |. F9 E  下面有22个不同的类(继承了SearchTerm类)供我们使用:0 B8 Y$ l% H! l3 W
  AND terms (class AndTerm)
0 i/ ~. _9 Y: D  S( q  OR terms (class OrTerm)7 V3 Q( C$ t) S( l  A
  NOT terms (class NotTerm). C, n1 E( [8 N8 b4 J
  SENT DATE terms (class SentDateTerm)
% W# ]/ ~& ?9 h  ]4 M0 v  CONTENT terms (class BodyTerm)
1 ^  ^! s$ J0 u* b: C. v  HEADER terms (FromTerm / FromStringTerm, RecipientTerm / RecipientStringTerm, SubjectTerm, etc.)* Q2 X2 ^3 u# O
  使用这些类定义的断语集合,我们可以构造一个逻辑表达式,并在Folder中进行搜索。下面是一个实例:在Folder中搜索邮件主题含有“ADV”字符串或者发信人地址为friend@public.com的邮件。& ^& U1 y5 F' E" |
  SearchTerm st = new OrTerm( new SubjectTerm("ADV:"), new FromStringTerm("friend@public.com"));Message[] msgs = folder.search(st);. {% K! i$ ?1 X7 A
  六、参考资源
4 L; z. L. x& f- F* m! C4 _9 }" u8 E  JavaMail API Home5 R: S, t: `0 I! K
  Sun’s JavaMail API基础
0 H: M7 ^' ^7 t6 q  JavaBeans Activation Framework Home# _+ m9 r+ g. @9 {$ @; p' |) f5 q
  javamail-interest mailing list
9 D7 C* W- G" h3 r! I2 k/ y* q  Sun's JavaMail FAQ
+ U# E' T+ ^1 t- u7 m5 i  jGuru's JavaMail FAQ( ]# N" J; ~3 [/ u4 V
  Third Party Products List
回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

Archiver|手机版|小黑屋|Woexam.Com ( 湘ICP备18023104号 )

GMT+8, 2024-5-3 05:36 , Processed in 0.244268 second(s), 33 queries .

Powered by Discuz! X3.4 Licensed

© 2001-2017 Comsenz Inc.

快速回复 返回顶部 返回列表