今天来介绍一下应用层的电子邮件服务,我们每天几乎都在用,电子邮件(email)服务也是一种基于C/S模式的服务,它采用的是一种"存储-转发"的服务,是一种异步通信方式的服务,可以进行非实时通信.
整个电子邮件系统包括"用户代理"(UA)和"消息传输代理"(MTA)两大部分.
用户代理(UA):为用户提供操作界面,位于客户端主机内.
信息传输代理(MTA):负责消息的传输,即所谓的"电子邮局",一般位于邮件服务器中.
收发电子邮件,需要遵守一些标准,常见的电子邮件协议有以下三个:
SMTP(简单邮件传输协议):提供电子邮件的发送服务.端口号25.
POP3(邮局协议3):提供邮件接受的服务.端口号110;
IMAP(互联网消息访问协议):一种功能更强大,也更复杂的邮件接收的协议.改进了一些POP3的不足.端口号143.
SMTP只能基于文本进行传输,不能直接传输图片,视频等非文字的信息,但是配合一个叫做MIME的辅助协议可以将二进制编码后再通过SMTP进行传输.
什么是抄送?
假设A发邮件给了B,并且抄送给了C,那么B可以看到,"哦,A给我发邮件了,但同时也发给了C,这邮件被C围观了".
什么是暗送?
同样,A发邮件给了B,并且暗送给了C,则B看不到邮件发给了C,"哦,A只给我发了邮件".
当发送方计算机和支持SMTP的邮件服务器建立连接后,把电子邮件准确无误的发送到接收方的邮件服务器,如果跨多个网络,SMTP可以跨网接力式传输.
注意:下面的不是指建立TCP连接,3个阶段都是在TCP已建立好连接的基础上进行的.
连接建立:TCP25号端口连接建立好以后,发件方调用SMTP客户端发送HELO命令,和自己的域名,向收件方表明身份,如果通过验证,收件方会返回"250 OK"的应答消息,这时就建立好了会话连接.
邮件传输:此阶段发送方会使用多个命令,如用MAIL FROM指示发件人的邮箱地址,使用RCPT TO指示收件人的邮箱地址,此命令可以发送多次,如果服务器准备接受邮件则回复"250 OK",这时候,发送方就可以使用DATA命令发送邮件内容了,发送完毕后,服务器依然会返回"250 OK"表示一切顺利.
连接释放:没有邮件传输时,发送方发送一条QUIT命令,请求关闭连接,服务器正常情况下会返回221的应答,表示接受关闭请求,释放本次连接.
类比SMTP的作用,当接收方计算机与支持POP3协议的邮件服务器建立连接后,把存储在该服务器上的邮件准确无误的下载到收件方的主机.POP3是POP协议的第三个版本,解决了POP服务器上用户阅读了某邮件后即删除该邮件的不足.
连接建立:建立好TCP110端口的连接后,进入身份验证阶段,收件方使用USER和PASS作为用户名和密码提供给服务器,完成身份验证.
邮件接收:通过身份验证后,进入事务处理阶段,收件方可以发送POP3命令进行相应操作,邮件服务器会接受命令并作出响应.
连接释放:操作完成,收件方发送QUIT命令,进入更新状态,服务器确认关闭连接,并更新邮件存储区.
最后在简单介绍一下IMAP协议和MIME协议:
IMAP,提供了更加全面的功能,允许用户像管理本地文件一样自由地组织自己的邮箱,提供邮件摘要功能使用户可有选择地下载邮件,还提供了邮箱共享的功能
IMAP4能在三种模式下工作:离线模式、在线模式和断线模式.
华丽的分割线之后,终于来到了我真正想写的部分,以前学JavaEE的时候,接触过javaMail发送邮件的功能,那时候对Email协议理解的很浅薄,很多知识点马马虎虎就过去了,看完了计算机网络的电子邮件协议,又回去重新复习了一下JavaMail的相关内容,这里做一个总结:
此session非彼session,这里是javaMail中的一个类,Session类定义了基本的邮件会话,如果你得到了Session,表示你已经和邮件服务器连上了,完成了第一步,连接建立的阶段,它的作用与JDBC的Connection有点相似.
通过查j2ee的文档,可以发现这样两个方法,1.Session.getInstance(Properties prop); 2.Session.getInstance(Properties prop, Authenticator auth);,该方法返回一个Session,但需要两个参数,先来看第一个参数Properties prop,这个参数需要指定两个键值对,第一个是指定服务器主机名,第二个是指定是否需要认证,可以这样得到:
Properties prop = new Properties();
prop.setProperty(“mail.host”, “smtp.qq.com”);//设置服务器主机名
prop.setProperty(“mail.smtp.auth”, “true”);//设置需要认证
第二个参数Authenticator是一个接口表示认证器,即校验客户端的身份。我们需要自己来实现这个接口,实现这个接口需要使用账户和密码。
是抽象类Message的实现类,表示一个邮件对象,可以调用它的set*()方法,设置发件人,收件人,主题,正文等等...
是抽象类Address的实现类,是一个邮件地址类,用来设置邮件的发件地址,收件地址,等.如Address address=new InternetAddress("[email protected]");
这个类实现了发送消息的协议,即SMTP,在发送消息时,使用该类的一个抽象方法send(Message msg);方法是多样的。当然,也可由Session获得相应协议对应的Transport实例。并通过传递用户名、密码、邮件服务器主机名等参数建立与邮件服务器的连接,并使用sendMessage()方法将信息发送,最后关闭连接.
下面通过代码实现发送邮件的过程:
/** * 第一种方式 * 使用Session.getInstance(Properties prop); * @throws Exception */ @Test public void sendMail1() throws Exception{ Properties prop=new Properties(); //指定一个默认的邮件服务器的主机名 prop.setProperty("mail.host", "smtp.163.com"); //设置smtp服务器需要进行验证 prop.setProperty("mail.smtp.auth", "true"); //指定一个默认的消息访问协议,Session的getTransport()返回实现此协议的Transport对象 prop.setProperty("mail.transport.protocol", "smtp"); //1.获取连接 Session session=Session.getInstance(prop); //2.通过Session得到Transport对象, Transport ts=session.getTransport(); //如果不设置上面的prop的第三个键值对,可以使用下面这种方式 //Transport ts=session.getTransport("smtp"); //3.登录邮件服务器,使用你的用户名和密码 ts.connect("smtp.163.com", "邮箱用户名", "密码"); //4.创建邮件内容Message MimeMessage msg=new MimeMessage(session); //设置发件人邮箱 msg.setFrom(new InternetAddress("[email protected]")); //指明收件人邮箱 msg.setRecipient(RecipientType.TO, new InternetAddress("[email protected]")); //指定邮件的标题 msg.setSubject("第一封简单邮件"); //邮件的文本内容 msg.setContent("你好啊!", "text/html;charset=UTF-8"); //发送 ts.sendMessage(msg, msg.getAllRecipients()); //关闭连接 ts.close(); } /** * 第二种方式 * 使用Session.getInstance(Properties prop, Authenticator auth); */ @Test public void sendMail2() throws Exception{ /** * 1.得到session */ // Properties是Session的属性对象,用于封装针对SMTP的一些常用属性 Properties props=new Properties(); //设置smtp服务器地址 props.setProperty("mail.host", "smtp.163.com"); //设置SMTP服务器是否需要用户认证,默认为false,设为true props.setProperty("mail.smtp.auth", "true"); Authenticator auth=new Authenticator() { @Override protected PasswordAuthentication getPasswordAuthentication() { //注意:下面的登录用户名是@前边的内容,如果你的账号是[email protected],只需要输入haha return new PasswordAuthentication("邮箱用户名", "密码"); } }; Session session = Session.getInstance(props,auth); /** * 2.创建MimeMessage */ MimeMessage msg=new MimeMessage(session); //设置发件人 msg.setFrom(new InternetAddress("[email protected]")); //设置收件人 msg.setRecipients(RecipientType.TO,"[email protected]"); //设置抄送 msg.setRecipients(RecipientType.CC,"[email protected]"); //设置主题 msg.setSubject("第二封简单邮件"); //设置内容 msg.setContent("我是邮件","text/html;charset=utf-8"); /** * 3.发送 */ Transport.send(msg); }
以上只能完成文本的发送,如果想要发送图片,视频等多媒体的内容的话,就要考虑使用附件的方式.
当发送包含附件的邮件时,邮件体就为多部件的形式.上面在发送文本时,setContent()方法直接设置了正文内容.发送带附件的邮件时,就需要设置正文内容为MimeMultiPart.
MimeMultiPart是一个部件的集合类,那么什么是部件呢?就是MimeBodyPart.他们之间的关系是这样:
发送附件的具体代码,也很简单,这里以附件是一张图片为例,只需要把上面代码中:
//设置内容 msg.setContent("我是邮件","text/html;charset=utf-8");
修改成下面这样既可:
MimeMultipart list=new MimeMultipart(); //创建第一个MimebodtPart,为正文 MimeBodyPart part1=new MimeBodyPart(); part1.setContent("这是一封包含附件的垃圾邮箱","text/html;charset=utf-8"); list.addBodyPart(part1); //创建第二个MimebodtPart,为附件 MimeBodyPart part2=new MimeBodyPart(); //设置附件内容 part2.attachFile(new File("e:/照片/me.jpg")); //设置显示的文件名 解决乱码问题 part2.setFileName(MimeUtility.encodeText("哎哟,不错哦.jpg")); list.addBodyPart(part2); msg.setContent(list);
javaMail的相关知识点,这篇博文,讲得也很详细,可以参考一下:http://www.cnblogs.com/xdp-gacl/p/4216311.html