收藏 分销(赏)

张孝祥Java邮件开发详解.doc

上传人:xrp****65 文档编号:8753035 上传时间:2025-03-01 格式:DOC 页数:16 大小:654KB 下载积分:10 金币
下载 相关 举报
张孝祥Java邮件开发详解.doc_第1页
第1页 / 共16页
张孝祥Java邮件开发详解.doc_第2页
第2页 / 共16页


点击查看更多>>
资源描述
第3课时 关于邮件的理论以及通过cmd telnet到服务器发送和接受邮件(这一课时很重要) 其实邮件服务器就类似于邮局,发送方发到自己选择的邮件服务器,然后邮件服务器发送到目标服务器,目标服务器接收邮件,然后目标去目标服务器接受邮件。这这个过程中必须注意,除了用户取邮件用的是POP3协议,发送邮件到自己选择的服务器和自己这边的服务器转邮件到目标服务器之间用到的协议都是SMTP协议。只有用户主动去取邮件的时候才是POP3协议。 每个邮箱就相当于我们取信的和发信的邮局提供的邮件箱一样。我们的每个申请的邮箱账号,其实就是一个邮件箱,在邮件服务器上建立了一个账号和空间,邮件会放着这个服务器给我们开的账户的空间里。 邮件服务器按通讯协议分为两种类型 SMTP服务器 POP3/IMAP服务器 发送邮件或者是将邮件转发给另外一个邮件服务器就是用SMTP协议。 如果用户去取邮件,使用的协议就是POP3协议 Smtp服务器端口号是:25 Pop3服务器端口号是:110 Smtp服务器登陆,首先需要先使用base64格式编码用户名和密码,然后登陆服务器时输入编码后的用户名与密码给服务器验证才会通过,而pop3服务器登陆是不需要使用base64编码的,直接输入原来的用户名和密码。 Telnet :25 第四课时 发送邮件的原理(相当重要) 一定要注意,其实在cmd中和smtp服务器交流的时候,邮件和服务器之间的关系。 在写邮件之前,需要先定义mail from<>和rcpt to<>,这个是专门和服务器打交道,由服务器识别的。而在写邮件的时候,邮件分两部分,一个是邮件头,一个是邮件内容(从‘data’表示内容开始,data之后是全部的内容,以一个‘.’ 表示邮件内容结束),内容部分的头部分又需要再写一个from:<>,这里面写邮件的发件人,可以是开始的mail from也可以是另外一个邮箱名字。这个邮件头,服务器是不检查的,是给邮件阅读程序看的,所以发出去之后,在客户端邮件阅读程序(outlook)中我们看到的发件人实际上是邮件头上面写的发件人,而不是登陆邮箱的发件人的邮箱。一定要注意这点。所以有的邮件中看到发件人的名字是unknown,就是因为邮件头中没有输入发件人的名字。内容需要写from, to ,subject, date, 然后才是正式的正文 其实就相当于我们自己写了一封信,但是信上面的发信人不是写自己的名字,而是别人的名字,冒充别人,收件人是没办法区分发件人的真实身份的。 如果是普通的客户端访问smtp服务器是需要用户名和密码的,而针对各个smtp服务器之间,是不需要用户名和密码的,否则的话,smtp服务器是不可能知道那么多smtp服务器用户名和密码的。 实际上sina或者是其他邮件服务器提供商(smtp server)都是分为两块的,也就是有两种服务器, 一种是专门给普通用户访问的邮件服务器, 一种是给其他邮件服务器供应商访问的邮件服务器。 如果是普通用户访问的邮件服务器,那么用户访问的时候就需要提供用户名和密码,而其他邮件服务器供应商访问的专门的邮件服务器,是不需要用户名密码的。直接访问。服务器提供商会将专门给其他邮件服务器供应商访问的邮件服务器加入到DNS中,这样当系统查到这个访问的地址来自于DNS的其他邮件服务器,就会认为对方是邮件服务器,允许它不需要服务器密码直接访问这个邮件服务器。而专给个人客户访问的邮件服务器则是不能够加入DNS的,否则其他邮件服务器访问时,有可能找到的是这个专用个人访问邮件服务器,那么就会需要其他邮件服务器输入用户密码,导致其他邮件服务器无法登陆,邮件服务器之间无法互相转发邮件。 由于垃圾邮件泛滥,所以,现在网站提供的smtp服务器都要求提供身份验证,这就要求你在客户端软件(outlook)中的账户配置的服务器选项中,发送邮件服务器栏目中选中我的服务器要求身份验证,这样,服务器验证了身份才会和帮用户发送邮件。 第5课时 解释JavaMail包 JavaMail API按功能分为以下三大类 1. 创建和解析邮件内容的API:Message类是创建和解析邮件的核心API, 它的实力对象代表一封电子邮件。 2. 发送邮件的API:Transport类是发送邮件的核心API类,它的实例对象代表实现了某个邮件接收协议的邮件发送对象,例如SMTP协议 3. 接收邮件的API: Store类是接收邮件核心API类,它的实例对象代表实现了某个邮件接收协议的邮件接收对象,例如POP3协议 Session类:是用于定义整个应用环境信息的,以及收集客户端与邮件服务器建立网络连接的会话信息。如邮件服务器的主机名,端口号,采用的邮件发送和接收协议等。 其实JavaMail是分为两部分,一部分是接口包,一部分是实现类包。Mailapi.jar其实就是一个接口包,里面是没有实现的,只有在使用的时候,就需要实现包了,这些就是smtp.jar, pop3.jar等 生成邮件的过程中,还使用了一个第三方包,,也就是JAF(JavaBeans Activation Framework. JavaBean激活框架)是一个专用的数据处理框架,用于封装数据,并为应用程序提供访问的和操作数据的接口。Javamail.jar使用了JAF.不过,如果使用的是jdk6,就不需要导入jaf.jar了,因为jdk6中包括了jaf的jar包。 第六课时 发送邮件的最简单的代码和理论 Transport和Store都是Service的子类 Service有一个代码connect API,就是连接远程服务器的。 public void connect(java.lang.String host, int port, java.lang.String user, java.lang.String password) Transport先连接connect(… ),然后发送send(msg, new Address[]{new InternetAddress(“收件人地址”)}),最后关闭close。 记住服务器一般都需要论证auth,这个属性在标准的javamail spec中是没有的。但是在它的doc中,可以查看com.sun.mail.smtp的定义,里面定义了sun公司对于邮件发送需要定义的属性进行了定义,这是sun对mail.api的实现。其他公司也可以改。所以,它是com.sun.打头的,不是标准。这个很重要。 如果需要打印信息,则使用以下 session.setDebug(true); 注意transport中有两个方法,一个是send,一个是sendMessage,不要用send,因为它是一个static方法,而sendMessage才是一般的transport实例发送message的,它是在connect之后再发送,然后关闭的。而send方法,是不需要使用connect的,因为它会内部调用connect的,所以,send和sendMessage之间的区别是一个send每次发送都会重新connect一次服务器,而sendMessage可以通过一次transport connect之后,连续发多个message,再关闭,这样减少了连接的时间。目前从sina发送到163是有问题的,这不是代码问题,而是服务器问题。从sina发到sina没有问题。 // sessionProps.put("mail.smtp.host", ""); // sessionProps.put("mail.host", "localhost"); // sessionProps.put("mail.store.protocol", "pop3"); sessionProps.put("mail.transport.protocol", "smtp"); // sessionProps.put("mail.user", "silencehill_wu"); // sessionProps.put("mail.password", "wugang"); // sessionProps.put("mail.debug", "true"); sessionProps.put("mail.smtp.auth", "true"); Transport transport = session.getTransport(); transport.connect("",25,"silencehill_wu", "wugang"); Message message = new MimeMessage(session); message.setText("tttttttttt"); message.setFrom(addr1); message.setSubject("just test "); Address[] addrs = new Address[]{addr2}; try{ // Transport.send(message,addrs); transport.sendMessage(message,addrs); }catch(Exception ex){ ex.printStackTrace(); return; } transport.close(); 第7课时 可以发现代码中很方便,可以在多个地方设置相同的配置,只要可以通过,就都可以使用。 比如创建message的时候需要收件人地址,不过这个只是一个地址,而transport.sendMessage(message,addrs);的时候,这里的addrs是一个地址数组,可以放入多个收件人。所以,发送给多个的时候,就不要在创建message的时候发送,而是在sendMessage的时候写入。 getInstance每次都会返回一个新的session对象,和上一个不同。 而getDefaultInstance如果没有,则创建一个新的,否则返回的是已经存在的session对象,不会创建新的。但是这个default的似乎有一个问题,就是如果使用这个方法发送之后,接着收就会出问题,也就是说同一个session不能发了再接着收,原因具体不明。 如果是在服务器上运行,sessionProps.put("mail.smtp.auth", "true");是不需要加的,因为服务器通过DSN来确认服务器,是不需要认证的,只有客户端(outlook等)才需要论证。 使用Transport类发送邮件的方法,一个过程只能发送一次邮件。由于是Transport静态API send内部包含了connect, send和close session的功能,所以,connect中的连接信息就需要分离出来。 添加以下代码 sessionProps.put("mail.host", "");//指定mail服务器地址 //下面使用了Authenticator来传入连接服务器的用户名和密码 Session session = Session.getInstance(sessionProps, new Authenticator(){ @Override protected PasswordAuthentication getPasswordAuthentication() { return new PasswordAuthentication("silencehill_wu","wugang"); } } ); //下面加入了多个收件人地址,使用InternetAddress.parse解析多个地址成为收件人数组。 message.setRecipients(RecipientType.TO, InternetAddress.parse("silencehill_wu@,wugang00001@,silencehill_wu@")); //不用setText,改用setContent,可以设置不同邮件不同的显示格式的内容 message.setContent("<span style='color:red'>你好吗,哈哈!</span>", "text/html;charset=GBK"); 第8课时 邮件的基本格式与编码 对于邮件中的非英文字符,我们往往需要特别的格式来发送邮件。当然和我们的字符编码是不一样的。 现在常用的两种编码方式,一种是base64和quote-printable。对于被发送的内容,必须保证所有的字符都被编码成一种网络可以识别的特殊格式,这样才都可以被网络传输,以免部分发送内容中的部分二进制或者特殊符号被网络过滤掉。 这些编码格式可以通过看邮件的属性,或者通过editplus的编辑就可以看到了。 比如 ------=_Part_374982_25774252.1288351532984 Content-Type: text/html; charset=gbk Content-Transfer-Encoding: quoted-printable <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.= w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> <html xmlns=3D"http://www.w3.org/1999/xhtml"> <head> <meta http-equiv=3D"Content-Type" content=3D"text/html; charset=3DGBK" /> <title>=C4=FA=D4=DA=B0=D9=BA=CF=D3=D0=D0=C2=CF=FB=CF=A2=C0=B2!</title> <style type=3D"text/css"> 如果要看base64的,只需要写一封邮件,在收件人中写入中文 silencehill_wu@,然后主题写haha,content邮件内容中写中文信息,然后设置编码为纯文版,保存成eml文件,用editplus编辑。就会看到 From: "john wu" <silencehill_wu@> To: =?gb2312?B?1tDOxCA8c2lsZW5jZWhpbGxfd3VAMTYzLmNvbT4=?= Subject: haha Date: Tue, 2 Nov 2010 00:02:22 +0800 MIME-Version: 1.0 Content-Type: text/plain; charset="gb2312" Content-Transfer-Encoding: base64 X-Priority: 3 X-MSMail-Priority: Normal X-Unsent: 1 X-MimeOLE: Produced By Microsoft MimeOLE V6.00.2900.5931 1tDOxNDFz6I= 其中的1tDOxNDFz6I=其实就是中文信息经过base64的传输编码处理之后的字符。 其实google搜索的时候,输入中文,点击搜索,也是用的编码方式也就是URL编码。 比如搜索中文两个字。.hk/search?hl=zh-CN&source=hp&biw=1588&bih=818&q=%E4%B8%AD%E6%96%87&aq=f&aqi=&aql=&oq=&gs_rfai= %E4%B8%AD%E6%96%87就是中文编码之后的样子。 可以在UltrEdit中,先输入中文,然后将Ascii改成utf-8,就会出现我们看到的二进制数据,可以发现‘%’被去掉了。所以%E4其实就是3个字节,一个字符一个字节,所以24个bit。 现在继续谈邮件的编码方式,因为有的邮件需要发送特殊的数据要转化成英文字符。比如说中文。或者某些不可以被传送的字符。 邮件中可以设置编码方式,在邮件option(选项)的发送tab中,邮件发送格式中先选中纯文本,然后点击”纯文本设置….”,里面会有一个列表选择编码方式,一个是base64,一个是可打印项目(quoted-printable),改了这个,就会保存成我们需要的邮件编码。Eml中的文字也会变成相应的编码格式字符。 详细介绍编码。 Base64编码:比如下面3个字节[0110,0001][0110,0010][0110,0011]如果想要编码怎么编?我们选了64个英文字符,是可以用6位bit来表示,也就是2的6次方(111111),刚好可以完全表示特定的64位字符(abcd....zABCD…Z)如果字符是35,就是对应的第35的字符是什么。这样可以将上面的3个字节拆分成4*6个字符的编码,这样3个字符就变成4个英文字符。扩大了三分之四倍。 quoted-printable编码:只对非ASCII字符的数据进行编码。如果是英文符号,我们不编码。每个非ASCII字符的自己数据,都被转化成’=’后跟着个自己的十六进制数据,比如说 ‘ab中国’的quoted-printable 编码就是ab=d6=d0=b9=fa这样ab没有变,而中国就从4个字节变成12个字节,扩大了3倍。(中就是d6d0, 国就是b9fa),如果主要是英文,很少中文,就可以用quoted-printable编码,如果很多中文,就用base64编码 第九课时 如果需要发送附件,首先需要在邮件option(选项)的发送tab中,邮件发送格式中选‘HTML’,这样才可以发送图片,超链接,附件等等。 其实即使放了附件和图片,实际上只是一个文件而已,类似于压缩文件一样,通过一定编码而成。 创建一个正常的邮件,里面放了正文,图片还有一个附件,然后保存。用ultraedit看里面的内容。可以看到它分为几个部分。 格式大致如下 From: "john wu" <silencehill_wu@> Subject: =?gb2312?B?1tDOxA==?= Date: Tue, 2 Nov 2010 23:01:33 +0800 MIME-Version: 1.0 Content-Type: multipart/mixed; boundary="----=_NextPart_000_001F_01CB7AE1.E7A45D10" X-Priority: 3 X-MSMail-Priority: Normal X-Unsent: 1 X-MimeOLE: Produced By Microsoft MimeOLE V6.00.2900.5931 This is a multi-part message in MIME format. ------=_NextPart_000_001F_01CB7AE1.E7A45D10 Content-Type: multipart/related; type="multipart/alternative"; boundary="----=_NextPart_001_0020_01CB7AE1.E7A45D10" ------=_NextPart_001_0020_01CB7AE1.E7A45D10 Content-Type: multipart/alternative; boundary="----=_NextPart_002_0021_01CB7AE1.E7A45D10" ------=_NextPart_002_0021_01CB7AE1.E7A45D10 Content-Type: text/plain; charset="gb2312" Content-Transfer-Encoding: base64 YWFhYmJiY2NjZGRk ------=_NextPart_002_0021_01CB7AE1.E7A45D10 Content-Type: text/html; charset="gb2312" Content-Transfer-Encoding: base64 PCFET0NUWVBFIEhUTUwgUFVCTElDICItLy9XM0MvL0RURCBIVE1MIDQuMCBUcmFuc2l0aW9uYWwv L0VOIj4NCjxIVE1MPjxIRUFEPg0KPE1FVEEgY29udGVudD0idGV4dC9odG1sOyBjaGFyc2V0PWdi MjMxMiIgaHR0cC1lcXVpdj1Db250ZW50LVR5cGU+DQo8TUVUQSBuYW1lPUdFTkVSQVRPUiBjb250 ZW50PSJNU0hUTUwgOC4wMC42MDAxLjE4OTc1Ij4NCjxTVFlMRT48L1NUWUxFPg0KPC9IRUFEPg0K PEJPRFkgYmdDb2xvcj0jZjRmNGY0Pg0KPERJVj48Rk9OVCBzaXplPTI+YWFhYmJiY2NjZGRkPElN RyBib3JkZXI9MCBoc3BhY2U9MCBhbHQ9IiIgYWxpZ249YmFzZWxpbmUgDQpzcmM9ImNpZDo3N0Mw QzQ3QkZERDc0MEU5QjYyNDdBRTFENzVBRjc0NUA1OGZhYWQ0OTZmNzE0ODYiPjwvRk9OVD48L0RJ Vj48L0JPRFk+PC9IVE1MPg0K ------=_NextPart_002_0021_01CB7AE1.E7A45D10-- ------=_NextPart_001_0020_01CB7AE1.E7A45D10 Content-Type: image/jpeg; name="=?gb2312?B?1tbK9y5KUEc=?=" Content-Transfer-Encoding: base64 Content-ID: <77C0C47BFDD740E9B6247AE1D75AF745@58faad496f71486>” 可以发现首先是multipart/mixed表示多格式组成,而里面又分为多个部分,每个部分是由boundary来设置,表示块分割,其中最后用—表示一块的结束。如果一块之中包含了其他的part,说明是嵌入其中的更小的块。 第11课时 填充和完善创建复杂邮件的细节代码 在创建MimeMultipart的时候,构造函数是一个String,指定的是subType,也就是说是multipart/mix中的mix这一部分,因为multipart是主类型。所以,subType就是只mixed 不论是附件还是内容,首先都是MimeBodyPart,在建立这些之前,先要给这个邮件建立一个MimeMultipart,它代表了整个MimeMessage的内部,包括附件和内容。所以首先建立MimeMultipart,然后用它加入两个部分一个是attach部分,一个是内容,都是MimeBodyPart。但是,实际上当内容中有一个是图片的时候,则需要在MimeBodyPart中加入一个MimeMultipart,这样才行。所以需要这个MimeBodyPart.setContent(MimeMultipart),然后又括号中的MimeMultipart来添加附件MimeBodyPart和文本content。 设置附件的时候,附件必须要被下载,而下载就需要附件名字,所以必须设置附件名。首先用英文名,中文暂时设置会有问题。例如: MimeBodyPart attach2 = new MimeBodyPart(); DataSource ds2 = new FileDataSource(new File("G:\\照片\\自己的照片\\myphoto\\SnapCap2.bmp")); DataHandler dh2 = new DataHandler(ds2); attach2.setDataHandler(dh2); attach2.setFileName("b.bmp"); msgMultipart.addBodyPart(attach2); 如果要设置邮件的body+其中的插图,就必须设置一个新的MimeMultipart MimeMultipart bodyMultipart = new MimeMultipart("related"); content.setContent(bodyMultipart); 这个bodyMultipart就用来包括邮件中的内容和图片的。而内容分为一个字符MimeBodyPart代表字符串和一个MimeBodyPart代表图片,而图片部分就类似于一般的附件,没有特别的配置,只需要bodyMultipart.add(图片bodyPart)就可以了。但是其实它是单独存在的,为了内容中的某个图片存在的,所以其中要多加入一句gifpart.setHeader("Content-Location", " 这里面的 对于需要显示的文本body,需要设置的是content,代码如下 MimeBodyPart htmlpart = new MimeBodyPart(); htmlpart.setContent("你们不是很牛吗,和我们比一比<img src=' "text/html;charset=gbk"); bodyMultipart.addBodyPart(htmlpart); 至于保存成一个文件,则需要使用MimeMessage的writeTo message.saveChanges(); FileOutputStream fos = new FileOutputStream(new File("c:\\test.eml")); message.writeTo(fos); fos.close(); 第12课时 中文附件名_回信地址_友好名称等技巧 如果是使用中文附件名或者是中文名字的收件人,回复人名称,都需要编码。具体编码的方式如例:To: =?UTF-8?B?5ZC06ZKi?= silencehill_wu@ 可以看到以=?开始,然后是charset字符集=UTF-8,然后又是?分割,后面的B表示Base64,如果是Q则是quote-printable。然后又是?分割,接着是编码后的字符(这里是吴钢进行编码),然后?=结束了。后面就是空格,然后就是真实的邮件地址了。这种格式=?UTF-8?B?5ZC06ZKi?=,我们只需要使用MimeUtility.encodeText(“张三”)就可以做到了。不需要我们手动编写字符。 如果想要收件人或着回复人显示的是用户名而不是地址,格式如下“张三” zhangsan@>,这样的格式看邮件的时候出来的就是显示张三,而地址是看不到的。对于中文名字同样是使用MimeUtility.encodeText String recname = MimeUtility.encodeText("张三"); message.setRecipient(RecipientType.TO, new InternetAddress("\""+ recname + "\" <zhangshan@>")); 如果想要设置收件人,可以设置单个地址也可以设置数组,setReceipt和setReceipts message.setRecipient(RecipientType.TO, new InternetAddress("\""+ recname + "\" <silencehill_wu@>")); 还可以设置点击回复之后的地址,比如 message.setReplyTo(InternetAddress.parse("silencehill_wu@, wugang00001@"));这里就是地址数组,可以用InternetAddress.parse转换字符成为邮件地址,然后用setReplyTo设置回复人。 至于附件如果是设置中文名,也是使用MimeUtility.encode转换编码,显示出来的附件格式就是这样的。代码如下: String str = MimeUtility.encodeText("开玩笑"); attach2.setFileName(str + ".bmp"); msgMultipart.addBodyPart(attach2); 产生的格式如下: Content-Type:application/octet-stream; name="=?UTF-8?B?5byA546p56yR?=.bmp" Content-Transfer-Encoding: base64 Content-Disposition:attachment; filename="=?UTF-8?B?5byA546p56yR?=.bmp" Encode会自动在字符中看英文还是中文多,如果英文多就是Q,中文多就是B 第13课时 如何直接发送一封现有的邮件文件 如果我们要发送一封固定内容和固定的收件人的复杂邮件,可以通过outlook先创建一个文件,将内容和收件人等都写好。然后保存成一个eml文件,这样就有一个现成的.eml了,那么可以直接使用这个eml文件,然后发送,记住,.eml中的发件人邮件人地址是不能直接使用的,因为发送人是根据真正发送邮件的用户地址来决定,不可以在eml中读,所以,需要用户重新用代码写发送email地址,而其他接收人地址则是原来eml中的。 第14课时 在Web应用中增加邮件发送功能 子类的异常不可以超过父类,只能少于子类。所以,有多出的异常需要使用try/catch,这样就可以了。 缺省情况下tomcat是没有javaee.jar,它只是web服务器,而不是j2ee或者应用服务器。而javamail是在jdk6的javaee中的。 发邮件实际上是需要异步处理,由一个独立的邮件发送模块来发送,过一段时间就找目录中文件,然后逐步进行发送,可以通过jms或者通过进程来实现。而不应该阻碍servlet处理,否则会导致网络阻塞。 第15课时 JNDI资源与类加载问题 Tomcat提供了jndi的方式来支持javamail的。 Tomcat在系统中配置JNDI,都是定义在$CATALINA_BASE/conf/server.xml或者每个web应用服务自己独自定义的文件META-INF/context.xml 1. 这其中的的属性的配置和配置session中的那些属性一致。 <Context> <Resource name="mail/Session" auth="Container" type="javax.mail.Session" mail.smtp.host="localhost" mail.smtp.auth="true" mail.transport.protocol="smtp"/> </Context> 2. 而获得session的方法如下 Context initCtx = new InitialContext(); Context envCtx = (Context) initCtx.lookup("java:comp/env"); Session session = (Session) envCtx.lookup("mail/Session"); Message message = new MimeMessage(session); 因为session的获得不是我们定义的,所以,它的初始化函数中需要Authenticator的那个,就不能使用了,而是需要让Transport来建立连接。Transport也不能直接使用send,而是要session.getTransport();这样的transport来手工建立连接 而这时候transport使用的是普通API sendMessage而不是静态API send,一定要注意。 代码如下: Transport transport = session.getTransport(); transport.connect("",25,"silencehill_wu", "wugang"); transport.sendMessage(message, to); transport.close(); 如果要使用tomcat的jndi来访问邮件服务,其实就是访问session。那么这个session的产生就是tomcat容器来产生,这个不是访问web应用的lib来获得的,而是需要在容器启动的时候就创建,所以,这就需要将mail.jar放在web容器的lib中。同样的是,关于连接数据源需要jndi访问的时候,需要驱动,也是需要放到容器lib中的,比如mysql connect driver。这样才能通过jndi访问数据库的jndi。 注意出现的问题 java.lang.ClassCastException: javax.mail.Session cannot be cast to javax.mail.Session 这是因为tomcat已经有一个容器,它启动了tomcat lib中mail.jar,而我们的web应用程序的lib中也包括了一个mail.jar,这两个虽然一样,但是使用的是不同的类加载器,在虚拟机中是不同的位置,容器启动认为这是两个类是不一样。所以,去掉web应用
展开阅读全文

开通  VIP会员、SVIP会员  优惠大
下载10份以上建议开通VIP会员
下载20份以上建议开通SVIP会员


开通VIP      成为共赢上传

当前位置:首页 > 教育专区 > 其他

移动网页_全站_页脚广告1

关于我们      便捷服务       自信AI       AI导航        抽奖活动

©2010-2026 宁波自信网络信息技术有限公司  版权所有

客服电话:0574-28810668  投诉电话:18658249818

gongan.png浙公网安备33021202000488号   

icp.png浙ICP备2021020529号-1  |  浙B2-20240490  

关注我们 :微信公众号    抖音    微博    LOFTER 

客服